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本 教材 按照 教 、 学 、 做 、 练 一 体 化 模式 精 编 了 网 络 编程 与 动态 网 站 制作 的 核心 内 容 , 以 知 
识 要 点 ,技能 操作 、 拓 展 训练 为 单元 组 织 教材 的 体系 结构 。 知 识 要 点 给 出 了 最 重要 和 最 实用 
的 知识 ,是 教师 需要 重点 讲授 的 部 分 ; 技能 操作 给 出 了 学 生 与 教师 共同 完成 的 操作 任务 ; 
拓展 训练 给 出 了 需要 学 生 独 立 完成 的 实践 活动 。 

全 书 共 分 为 12 章 。 第 1 章 主要 介绍 网 络 编程 与 动态 网 站 制作 的 基础 知识 ,重点 讲述 客户 
端 开发 技术 HTML 超 文本 标记 语言 .CSS J FE sk JavaScript 脚本 语言 。 第 2 章 主要 介绍 
服务 器 端 开发 技术 JSP 的 初步 知识 ,包括 JSP 动态 网 站 开发 环境 的 搭建 及 利用 文本 编辑 器 、 
IDE 集成 开发 环境 开发 简单 Web 应 用 的 基本 步骤 。 第 3 章 介 绍 JSP 基本 语法 ,包括 JSP 文件 
的 构成 .成员 变量 和 方法 的 定义 Java 动态 元 素 以 及 常用 的 JSP 指令 标记 和 JSP 动作 标记 。 第 
4 章 介绍 常用 的 JSP 内 置 对 象 , 包 括 request, response, session, application 及 其 常用 方法 。 第 5 
章 介 绍 JavaBean 的 使 用 ,包括 编写 和 使 用 JavaBean、 获 取 和 设置 bean 属性 等 。JSP 和 JavaBean 
技术 的 结合 不 仅 可 以 实现 数据 表示 和 数据 处 理 分 离 ,而 且 还 可 以 提高 JSP 程序 代码 的 重用 度 ， 
是 JSP 编程 中 常用 的 技术 。 第 6 章 介绍 JSP 如 何 通过 JDBC-ODBC 桥接 方式 和 纯 Java 数据 库 
驱动 程序 方式 实现 对 数据 库 的 访问 ,如 Microsoft Access, SQL Server, MySQL, Oracle 等 数据 
库 。 第 7 章 和 第 8 章 讲 述 Servlet 的 运行 原理 以 及 基于 Servlet 的 MVC 模式 应 用 。 第 9 章 介绍 
Web 应 用 过 滤器 的 基本 概念 .运行 原理 及 其 实际 应 用 。 第 10 章 介绍 表达 式 语 言 EL 的 基本 用 
法 ,包括 使 用 EL 访问 对 象 的 属性 .EL 内 置 对 象 等 。 第 11 章 介绍 标准 标签 库 JSTL 的 基本 用 法 ， 
包括 普通 标签 .条件 控制 标签 .迭代 标签 等 的 使 用 。 第 12 章 通过 一 个 动态 网 站 开发 综合 实例 讲 
解 如 何 采用 JSP--JavaBean-- Servlet 来 开发 Web 应 用 项 目 。 另 有 附录 A 为 HTML 标签 库 ,附录 
BA HTML 颜色 库 ,附录 C 为 JSP 内 置 对 象 及 其 常用 方法 。 

本 书 内 容 翔实 丰富 ,注重 实践 操作 ,突出 能 力 培养 ,适合 作为 高 等 院 校 计算 机 相关 专业 
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第 1 章 


网 络 编程 与 动态 网 站 制作 基础 


网 络 编程 与 动态 网 站 制作 通常 包含 客户 端 开 发 和 服务 器 端 开发 两 部 分 : 客户 端 开发 主 
要 用 于 描述 信息 的 内 容 ,也 就 是 浏览 器 中 显示 的 网 页 ; 服务 器 端 开发 主要 用 于 实现 信息 的 
处 理 ,也 就 是 前 后 台 的 动态 交互 。 常 用 的 客户 端 开发 技术 有 HTML, CSS, HAR E E, A 
用 的 服务 器 端 开发 技术 有 ASP、JSP、PHP 等 。 本 章 主 要 介绍 客户 端 开发 技术 ,后 续 章节 将 
详细 介绍 服务 器 端 主流 开发 技术 JSP 的 相关 内 容 。 


1.1 HTML 


1.1.1 知识 要 点 


HTML(Hyper Text Markup Language, 超 文本 标记 语言 ) 是 一 种 用 于 描述 网 页 的 语 
言 , 它 不 是 编程 语言 ,而 是 标记 语言 。 换 言 之 ,HTML 使 用 标记 标签 来 描述 网 页 , 它 不 需要 
编译 ,可 以 直接 由 浏览 器 执行 。HTML 标签 是 由 尖 括 号 包围 起 来 的 关键 词 ,例如 过 html>， 
标签 通常 是 成 对 出 现 的 ,例如 二 body 二 和 三 /body 二 ; 标签 对 中 的 第 一 个 标签 是 开始 标签 ， 
第 二 个 标签 是 结束 标签 ,也 有 个 别 标签 独立 出 现 , 如 过 br 之 和 过 p 之 ; HTML 标签 可 以 有 一 
个 或 多 个 属性 ,属性 的 主要 用 途 是 辅助 HTML 标签 来 更 美观 地 、 更 准确 地 描述 网 页 的 内 
容 , 例 如 二 body bgcolor= "cyan" — rB H} bgcolor 属性 就 描述 了 网 页 的 背景 颜色 为 青绿 色 。 

HTML 标签 和 纯 文本 共同 构成 HTML 文档 , 即 通 常 所 称 的 静态 网 页 。 客 户 端 的 浏览 
器 能 够 读 取 HTML 文档 ,并 以 网 页 的 形式 显示 出 它们 。 浏 览 器 不 会 显示 HTML 标签 ,而 
是 使 用 标签 来 解释 页 面 的 内 容 。 

本 书 附录 A 给 出 了 常用 的 HTML 标签 ,附录 B 给 出 了 常用 的 HTML 颜色 库 , 供 参考 。 


1.1.2 技能 操作 


运用 HTML 语言 制作 简单 的 网 页 ,具体 任务 如 下 : 

编写 一 个 网 页 ,要 求 网 页 的 标题 为 “公安 网 站 示例 ”, 网 页 背景 是 灰色 ,解释 执行 后 在 浏 
览 嚣 中 显示 两 行文 字 一 一 “两 人 被 困 事故 车 中 当地 消防 急速 营救 "和 “组 织 党 员 干 部 开展 反 
腐 倡 廉 警 示 教 育 ”, 并 且 显 示 一 张 图 片 。 具 体 的 网 页 效果 如 图 1-1 所 示 。 

完成 任务 的 步 又 如 下 : 

* 编写 ,保存 HTML 文件 。 
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”使 用 浏览 器 打开 网 页 。 

1. 使 用 文本 编辑 器 编写 .保存 HTML 文件 

首先 需要 打开 一 个 文本 编辑 器 。 如 果 是 Windows 操作 系统 ,可 以 通过 “开始 ”一 “程序 ”一 
“附件 ”>“ 记 事 本 ”命令 来 直接 打开 一 个 文本 编辑 器 ; 如 果 是 其 他 操作 系统 ,可 以 参考 操作 
系统 的 帮助 手册 打开 一 个 文本 编辑 器 。 
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图 1-1 HTML 网 页 示例 


1) 编写 HTML 文件 的 内 容 
在 打开 的 文本 编辑 器 中 输入 下 面 的 内 容 : 


<html> 
<head> 
<title> A % IU 28 B| < / title 
</head> 
<body bgcolor= "gray"> 
Chl SHABKHI KAEH 354 b DI 8 8 3k < /h1> 
组 织 党 员 干 部 开展 反腐 倡 廉 警 示 教 育 
<p><!--#B--> 
<img src=" K KAH .jpg"> 
</body> 
</html> 


注意 事项 : HTML 文件 对 于 大 小 写 没 有 严格 限制 ,可 以 根据 个 人 习惯 自由 选择 。 
代码 说 明 : 二 html 写 与 二 /html 志 标签 限定 了 文档 的 开始 点 和 结束 点 ,在 它们 之 间 的 部 
分 是 文档 的 头 部 和 主体 。 正 如 上 面 代码 所 示 , 文 档 的 头 部 由 过 head 之 和 所 /head 二 标签 定 
义 , 而 主体 由 过 body 之 和 志 /body 二 标签 定义 。 在 文档 的 头 部 ,可 以 定义 网 页 标题 等 信息 ， 
如 <<title 之 和 到 /title> 标 签 就 指明 了 该 网 页 的 标题 是 “公安 网 站 示例 *。 在 文档 的 主体 部 
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分 ,可 以 定义 网 页 的 核心 内 容 , 如 文本 、 超 链接 、 图 像 表 格 和 列表 等 。 本 例 中 只 包含 了 文本 
MEH ,并 且 用 二 hl 之 和 二/hl 二 标签 定义 了 “两 人 被 困 事 故 车 中 ”当地 消防 急速 营救 ” 文 
本 为 最 大 标题 。 在 html 中 ,二 hl ~ 过 h6 二 标签 可 用 于 定义 标题 ,二 hl 二 定义 最 大 的 标 
题 ,二 h6 过 定义 最 小 的 标题 。html 标记 还 可 以 有 属性 ,这 些 属性 位 于 标记 的 开始 标签 中 , 例 
如 body 标签 表示 网 页 的 内 容 , 可 以 使 用 bgcolor 属性 设置 网 页 的 背景 颜色 ; <img E AE 
示 插 入 图 片 , 可 以 使 用 src 属性 设置 图 片 的 路 径 和 名 称 。HTML 中 也 可 以 有 注释 ,例如 
二 !-- 换 段 -- 二 就 是 一 种 说 明 性 的 注释 ,注释 不 显示 在 最 终 的 网 页 中 ,只 是 用 于 帮助 程序 员 
阅读 代码 。 

2) 保存 HTML 文件 

将 编辑 好 的 HTML 文件 保存 到 磁盘 的 目录 中 ,例如 保存 到 D:\ch01 目录 中 。 文 件 名 
“公安 网 站 示例 . html”, 其 中 . html 是 文件 的 扩展 名 。 使 用 记事 本 编辑 HTML 文件 ,在 保存 
时 ,必须 将 “保存 类 型 "选择 为 “所 有 文件 ”, 将 “编码 ”选择 为 ANSI。 如 果 保 存 HTML 文件 
时 ,计算 机 总 是 给 文件 名 末尾 额外 加 上 . txt, 那 么 在 保存 HTML 文件 时 可 以 用 双 引 号 括 起 
来 ,如 图 1-2 所 示 。 


文件 名 (N): “公安 网 站 示例 html" ` 
保存 类 型 中: | 所 有 文件 A 
O emk SBO: [ANSI - 取消 


1-2 记事 本 保存 文件 名 


2. 使 用 浏览 器 查看 网 页 

打开 浏览 器 ,查看 网 页 。 常 用 的 浏览 器 有 IE、FireFox 等 。 本 书 使 用 Windows 系统 自 
带 的 IE 浏览 器 (8.0 版 本 ) 浏 览 网 页 ,如 果 要 使 用 其 他 的 浏览 器 ,请 参考 相关 的 文档 。 打 开 
网 页 的 方法 如 下 : 

CD 打开 HTML 文件 所 在 的 文件 夹 (D:\ch01), 用 鼠标 双击 “公安 网 站 示例 . html” X 
件 , 系 统 将 启动 默认 的 浏览 器 ,打开 这 个 超 文本 文件 ,网 页 效果 与 图 1-1 相同 。 需 要 注意 的 
是 ,如 果 系 统 安装 了 其 他 的 浏览 器 ,并 且 默 认 的 浏览 器 不 是 正 , 那 么 可 以 使 用 下 面 的 方法 用 
IE 浏览 器 查看 网 页 。 

(2) 先 打开 下 浏览 器 ,然后 选择 “文件 >“ 打开 ”命令 ,在 打开 对 话 框 中 输入 文件 路 径 *D:\ 
ch0l\ 公 安 网 站 示例 . html”, 最 后 单 击 “ 打 开 ” 对 话 框 中 的 “确定 ”按钮 ,网 页 效果 与 图 1-1 相同 。 


1.1.3 拓展 训练 
1. HTML 文件 以 ( 。 ”) 标 签 对 开始 和 结束 。 
A. =body>…=/body> B. <head> ---< /head> 


C. <html>…</html> D. <title>---</title> 
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2. HTML 标签 的 属性 位 于 ( a 


A. 开始 标签 B. 结束 标签 

C. 开始 标签 和 结束 标签 之 间 D. A 和 B 
3. XF <title> -</title> HRE HAA ,错误 的 是 ( Ís 

A. 网 页 正文 的 开始 B. 用 于 显示 当前 网 页 的 标题 

C. fF <body>- </body> Z[ü] D. 44 F <head> -</head> Z [B] 
4. 用 来 将 文本 设置 为 最 大 标题 的 HTML 标签 是 ( ; 

A. <img>…</img> B. <h6>---</h6> 

C. <html>…</html> D. <h1>---</h1> 


5. 制作 一 个 简单 的 HTML 网 页 ,标题 为 “个 人 网 页 ”, 要 求 在 浏览 器 中 显示 你 的 姓名 、 
籍贯 .出 生日 期 ,所 在 院 校 .所 学 专业 .所 学 课程 .特长 爱好 等 相关 信息 ,为 了 增加 页 面 的 美观 
效果 可 以 设置 背景 颜色 .插入 图 片 、 建 立 超 链接 等 。 


1.2 CSS 


1.2.1 知识 要 点 


CSS(Cascading Style Sheets, 层 全 样式 表 ) 是 一 系列 格式 规则 ,用 于 控制 网 页 的 外 观 ， 
与 HTML 类 似 , 它 也 是 一 种 标记 语言 ,不 需要 编译 ,直接 由 浏览 器 执行 。 

CSS 在 网 页 制作 中 被 广泛 应 用 ,可 以 控制 许多 仅 使 用 HTML 无 法 控制 的 属性 。 例 如 ， 
可 以 为 所 选 文本 指定 不 同 的 字体 大 小 和 单位 (像素 、 磅 值 等 ); 通过 使 用 CSS 以 像素 为 单位 
设置 字体 大 小 ,还 可 以 确保 在 多 个 浏览 器 中 以 更 一 致 的 方式 处 理 网 页 布局 和 外 观 效果 。 另 
外 ,使 用 CSS 设置 网 页 格式 时 ,内 容 与 表现 形式 是 相互 分 开 的 。 网 页 内 容 即 HTML 代码 位 
于 自身 的 HTML 文档 中 ,而 定义 代码 表现 形式 的 CSS 规则 位 于 HTML 文档 的 另 一 部 分 
(通常 为 二 head 二 部 分 ) 中 或 男 一 个 文档 即 外 部 样式 表 中 。 如 此 一 来 ,网 页 的 外 观 修改 和 内 
容 更 新 就 变 得 相对 容易 ,从 而 有 利于 网 站 的 升级 和 维护 。 


1.2.2 技能 操作 


运用 CSS 控制 网 页 外 观 , 具 体 任 务 如 下 : 

将 如 图 1-1 所 示 网 页 中 的 第 一 行文 字 即 “两 人 被 困 事 故 车 中 当地 消防 急速 营救 "的 显 
示 外 观 修改 为 16 像素 、 红 色 、 粗 体 样式 ,效果 如 图 1-3 所 示 。 

完成 任务 的 方法 有 如 下 两 种 : 

。 通过 内 部 样式 表 控 制 网 页 外 观 。 

* 链 入 外 部 样式 表 控 制 网 页 外 观 。 

1. 通过 内 部 样式 表 控 制 网 页 外 观 

在 HTML 文件 的 二 head 二 区 域 里 可 以 直接 编写 样式 表 , 并 借助 二 style 二 标签 将 样式 
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DO Prem x 


de Ermi ANARE Y 


| 两 人 被 困 事故 车 中 当地 消防 急速 营救 


| 组 织 党 员 干 部 开展 反腐 倡 廉 警示 教育 


1-3 CSS 控制 网 页 外 观 


表 应 用 于 网 页 ,这 种 方式 可 以 称 为 内 部 样式 表 , 使 用 该 方式 完成 任务 具体 步骤 如 下 。 
1) 修改 HTML 文件 的 内 容 


<html> 
<head> 

<title> 4x 22 AAR — / title 

<style type="text/css"> 

«t-- 

hl{ 

font-size:16 pixels; 

color:red; 

font-weight: bold ; 

) 

E 

</style> 

</head> 
<body bgcolor=" gray" > 
二 hl 二 两 人 被 困 事故 车 中 ”当地 消防 急速 营救 二 /hl 二 
组 织 党 员 干 部 开展 反腐 倡 廉 警 示 教育 

«p 
<img src— "X X Ë] .jpg" 

</body> 

</html> 


注意 事项 : 有 些 低 版 本 的 浏览 器 不 能 识别 style 标记 ,这 意味 着 低 版 本 的 浏览 器 会 忽略 
style 标记 里 的 内 容 , 并 把 style 标记 里 的 内 容 以 文本 直接 显示 到 页 面 上 。 为 了 避免 这 样 的 
情况 发 生 ,可 以 用 加 HTML 注释 的 方式 (二 ! -- 注 释 -- 二 ) 隐 藏 内 容 而 不 让 它 显示 。 

代码 说 明 : CSS 格式 规则 由 两 部 分 组 成 即 选 择 器 和 声明 。 选 择 器 是 标识 已 设置 格式 元 
素 ( 如 p、hl、 类 名 称 或 id 等 ) 的 术语 ,而 声明 则 用 于 定义 样式 元 素 。 上 述 代码 中 定义 了 一 个 
样式 hl (hl 称 为 选择 器 ) ,在 hl 中 声明 了 三 个 属性 (用 大 括号 括 起 来 的 部 分 ) : font-size 属 
性 描述 字体 的 大 小 , 值 为 16 pixels( 像 素 ); color 属性 描述 字体 颜色 , 值 为 red( 红 色 ) ,font- 
weight 属性 描述 字体 粗细 , 值 为 bola HLK), 
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2) 浏览 网 页 效果 

打开 浏览 器 ,查看 网 页 ,具体 效果 与 图 1-3 相同 。 

2. 链 人 外 部 样式 表 控 制 网 页 外 观 

将 样式 表 保 存 为 一 个 独立 的 CSS 文件 ,然后 在 HTML 文档 中 用 二 link 二 标签 链接 到 这 
个 样式 表 文 件 , 这 种 方式 可 以 称 为 外 部 样式 表 , 使 用 该 方式 完成 任务 具体 步骤 如 下 。 

1) 编写 CSS 样式 表 文 件 

hl { 

font-size:16 pixels; 


color: red; 
font-weight: bold ; 


将 上 面 的 样式 表 保存 为 一 个 独立 的 CSS 文档 ,可 命名 为 01. css, 
2) 修改 HTML 文件 的 内 容 
<html> 
<head> 
<title> Z: % R] 3h 28 B| < /title> 
<link rel="stylesheet" type="text/css" href= "01.css"> 
</head> 
<body bgcolor= "gray"> 
<h> Bl A, Bt A GE 3 h 24 Hb qñ Bi 2 ë ## </> 
组 织 党 员 干 部 开展 反腐 倡 廉 警 示 教 育 
<p> 
<img src=" KË] Hr jpg" 
</body> 
</html> 
3) 浏览 网 页 效果 


打开 浏览 器 ,查看 网 页 ,具体 效果 与 图 1-3 相同 。 
1.2.3 拓展 训练 


1. 层 琶 样式 表 文 件 的 扩展 名 为 ( )。 
A. .html € 7 txt D; seaa 
2. 编写 一 个 HTML 网 页 ,用 来 宣传 你 所 就 读 的 学 校 , 并 借助 CSS 内 部 样式 表 和 外 部 
样式 表 控制 外 观 显 示 。 


1.3 JavaScript 


1.3.1 知识 要 点 


HTML 和 CSS 能 很 好 地 显示 数据 ,但 是 不 能 动态 地 处 理 数据 ,例如 进行 简单 的 四 则 运 
算 ,格式 验证 等 。 要 想 在 网 页 里 实现 动态 处 理 数据 ,可 以 借助 JavaScript。 
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JavaScript 于 1995 年 由 Netscape 公司 在 网 景 导 航 者 浏览 器 上 首次 设计 实现 而 成 。 
为 当时 Netscape 公司 与 Sun 公司 有 合作 ,Netscape 管理 层 希 望 它 外 观看 起 来 像 Sun 公司 
的 Java 语言 ,因此 取 名 为 JavaScripts JavaScript 实际 上 是 一 种 嵌入 在 网 页 中 的 脚本 语言 ， 
由 浏览 器 解释 执行 。 使 用 JavaScript 能 够 动态 地 改变 网 页 的 内 容 , 处 理 用 户 和 浏览 器 产生 
的 事件 。 目 前 , 数 百 万 计 的 网 页 使 用 JavaScript 来 实现 改进 设计 、 验 证 表单 等 功能 。 


1.3.2 技能 操作 


运用 JavaScript 对 页 面 数 据 进 行动 态 处 理 ,具体 任务 如 下 : 

在 网 页 中 对 用 户 输入 的 手机 号 码 格式 进行 动态 验证 : 如 果 用 户 输入 手机 号 码 位 数 不 足 
11 位 , 则 单 击 “提交 ?按钮 后 给 出 提示 ”手机 号 必须 是 11 位 ”; 如 果 用 户 输入 手机 号 码 为 非 数字 
字符 , 则 单 击 “ 提 交 ” 按 钮 后 给 出 提示 “手机 号 只 能 是 数字 ”, 具 体 效 果 如 图 1-4 和 图 1-5 所 示 。 


. [NP 
OB e-e] 


|| WE unas Blmnesue v 


FASE (1 他 数字 ) ， 


来 自 网 页 的 消息 


Y» Biogr Sanane v | 


手机 号 码 〈11 位 数字 ) : [135411abede 提交 


ES 


15 手机 号 码 字符 格式 验证 
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完成 任务 的 方法 有 如 下 两 种 : 

。 在 网 页 中 直接 嵌入 JavaScript 代码 。 

* 引入 JavaScript 文件 方式 。 

1. 直接 嵌入 方式 

通常 ,在 网 页 中 直接 嵌入 JavaScript 程序 代码 的 一 般 格 式 如 下 : 


<script language= "text/javascript"> 


JavaScript 程序 代码 
</script> 


其 中 的 type 属性 表示 脚本 的 类 型 ,其 值 必须 为 text/javascript。JavaScript 程序 代码 位 
T script RI /script > Wi Ag 4& < [8] ,这 些 代码 由 浏览 器 解释 执行 。 使 用 这 种 方式 对 用 
户 输入 的 手机 号 码 格式 进行 动态 验证 具体 步 又 如 下 。 

1) 编写 HTML 文件 “验证 手机 号 码 1. html" 


<html> 
<head> <title> Js WrüE-- JUI SUE WITA BR — / title — / head 
<body> 
<form name= forml target="_blank" method= post onsubmit= "return dosubmit(this) "> 
手机 号 码 (11 位 数字 ) : <input type="text" name= "mem_id"> 
<input type="submit" name 一 "submitl" value 一 "提交 "二 
</form> 
<script language=" text/javascript "> 
function dosubmit(frm) 
{ if(frm. mem id. value. length ! — 11) 
( alert(" 手 机 号 必须 是 11 位 "); 
return false; 
) 
else 
Í var mem value = frm.mem id. value; 
for(var i—0; i<mem_value. length; i++) 
{ if(mem_value.charAt(i)<'0' | | mem_value.charAt(i)>>'9') 
{ alert(" 手 机 号 只 能 是 数字 "); 
return false; 
) 
) 
) 
frm. submit() ; 
return true; 
) 
</script> 
</body> 
</html> 
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代码 说 明 : 位 于 二 script language=" text/javascript "> #ll< /script > WARI Z [8] WI 
部 分 为 JavaScript 程序 代码 。 在 这 部 分 代码 内 部 定义 了 一 个 名 为 dosubmit 的 函数 ,其 核心 
功能 是 判断 用 户 在 文本 框 中 输入 的 手机 号 码 是 否 为 11 位 数字 字符 ,如 果 不 符合 要 求 ,就 会 
弹出 相应 的 提示 对 话 框 。 
2) 浏览 网 页 效果 
打开 浏览 器 ,查看 网 页 ,效果 与 图 1-4 和 图 1-5 相同 。 
2. JavaScript 文件 方式 
JavaScript 文件 方式 是 把 JavaScript 程序 代码 写 在 一 个 单独 的 文件 中 ,然后 通过 script 
标记 把 这 个 JavaScript 文件 引入 到 网 页 中 。 在 网 页 中 引入 的 JavaScript 文件 的 具体 格式 
如 下 : 
script type= "text/javascript" src= "JavaScript 文件 的 URL"> 
</script> 
其 中 src 属性 表示 要 引入 的 JavaScript 文件 ,其 值 为 JavaScript 文件 的 URL, 3x 4 
URL 既 可 以 是 相对 地 址 ,也 可 以 是 绝对 地 址 。JavaScript 程序 代码 位 于 JavaScript 文件 中 ， 
这 些 代 码 由 浏览 器 解释 执行 。 需 要 注意 的 是 ,如 果 script 标记 使 用 src 属性 引入 了 一 个 外 
部 的 JavaScript 文件 ,那么 这 个 标记 的 起 始 标签 和 结束 标签 之 间 的 JavaScript 代码 将 被 忽 
略 ,不 会 被 执行 。 使 用 这 种 方式 对 用 户 输入 的 手机 号 码 格式 进行 动态 验证 的 具体 步骤 如 下 。 
1) 编写 JavaScript 文件 01. js 
function dosubmit(frm) 
t if(frm. mem id. value. length ! = 11) 
{ alert(" 手 机 号 必须 是 11 位 "); 
return false; 
) 
else 
{ var mem value = frm. mem id. value; 
for(var i—0; i<mem value. length; i++) 
{ if(mem value. charAt(i) '0' || mem value. charAt(i)7 '9") 
{ alert(" 手 机 号 只 能 是 数字 "); 
return false; 
) 
) 
) 
frm. submit() ; 


return true; 


) 
2) 编写 HTML 文件 “验证 手机 号 码 2. html" 


<html> 
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head <title>Js 验证 手机 位 数 是 否 符合 要 求 </title><</head> 

<body> 
<form name= "form1" target="_blank" method= post onsubmit= "return dosubmit(this) "> 
手机 号 码 (11 位 数字 ): <input type="text" name= "mem_id"> 


<input type="submit" name= "submit1" value 二 "提交 " > 
</form> 
<script language= "javascript" src= "01. js"> 
</script> 
</body> 
</html> 
3) 浏览 网 页 效果 


打开 浏览 器 ,查看 “验证 手机 号 码 2. html” 网 页 ,效果 与 图 1-4 和 图 1-5 相同 。 
1.3.3 拓展 训练 
1. 在 HTML 文件 中 ,JavaScript 程序 代码 位 于 ( Je 
A. <script> HIE /script>— Z lš] B. <code> #ll< /code> Z lš] 
C. <javascript>#I< /javascript^zt|B] D. <script> R% Pj 
2. script 标记 的 type 属性 值 为 ( ) 
A. javascript B. text C. javascript/text D. text/javascript 
3. 已 知 JavaScript 文件 tom.js, 文 件 内 容 为 : 
alert("tom file") ; 


并 且 在 一 个 HTML 文档 中 有 下 面 的 代码 : 


<script type= "text/javascript" sre= "tom. js"> 
alert("abc"); 


</script> 
H IE WE RTIA HTML 文档 ,将 弹出 一 个 对 话 框 ,对 话 框 中 显示 ( O. 
A. tom file abc B. tom file C. abc D. 什么 也 不 显示 


4. 将 “验证 手机 号 码 2. html" 文 件 中 的 代码 : 

<script type= "text/javascript" src= "01. js" — /script2> 

改 为 : 

<script type= "text/javascript" src="01.js"> 

alert(" 我 被 忽略 了 ,没有 执行 !!"); 

</script> 

保存 文件 ,然后 用 浏览 器 打开 该 文件 查看 效果 ,观察 “我 被 忽略 了 ,没有 执行 11” 这 句 话 
是 否 显示 在 网 页 中 。 
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1.4 小 结 


网 络 编程 与 动态 网 站 制作 通常 包含 客户 端 开发 和 服务 器 端 开发 两 部 分 : 客户 端 开 
发 主要 用 于 描述 信息 的 内 容 , 也 就 是 浏览 器 中 显示 的 网 页 ; 服务 器 端 开发 主要 用 于 
实现 信息 的 处 理 , 也 就 是 前 后 台 的 动态 交互 。 常 用 的 客户 端 开发 技术 有 HTML、 
CSS JavaScript 脚本 语言 等 。 

HTML(Hyper Text Markup Language, 超 文本 标记 语言 ) 是 一 种 用 于 描述 网 页 的 
语言 , 它 不 是 编程 语言 ,而 是 标记 语言 。 

CSS(Cascading Style Sheets, 层 至 样式 表 ) 是 一 系列 格式 规则 ,用 于 控制 网 页 的 外 
W, HTML 类 似 , 它 也 是 一 种 标记 语言 ,不 需要 编译 ,直接 由 浏览 器 执行 。 
JavaScript 是 一 种 嵌入 在 网 页 中 的 脚本 语言 ,由 浏览 器 解释 执行 。 使 用 JavaScript 
能 够 动态 地 改变 网 页 的 内 容 , 处 理 用 户 和 浏览 器 产生 的 事件 。 
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JSP(Java Server Pages, Java 服务 器 端 页 面 ) 是 由 Sun Microsystems 公司 ( 现 已 被 
Oracle 公司 收购 ) 倡 导 多 家 公司 共同 参与 建立 的 一 种 动态 网 页 技术 标准 。 需 要 说 明 的 是 ， 
这 里 提 到 的 “动态 ”, 并 不 是 指 放 在 网 页 上 的 动态 图 片 和 动画 效果 ,而 是 指 动态 交互 , 即 网 页 
会 根据 用 户 的 要 求 和 选择 而 动态 改变 和 响应 。 从 本 章 开 始 将 详细 介绍 利用 JSP 技术 进行 
网 络 编程 与 动态 网 站 制作 的 具体 内 容 。 


2.1 搭建 开发 平台 


2.1.1 知识 要 点 


1. JSP 简介 

JSP 技术 是 在 传统 的 HTML 网 页 文件 中 插入 Java 程序 段 和 JSP 标记 ,从 而 形成 JSP 
文件 来 创建 动态 页 面 的 一 种 方法 , 它 使 得 Web 服务 器 .应 用 程序 服务 器 .浏览 器 以 及 各 类 开 
发 工具 环境 下 的 动态 网 站 开发 更 为 便捷 。 因 其 具有 简单 易 用 、 平 台 无 关 、 安 全 可 靠 等 特点 ， 
目前 已 成 为 动态 网 站 制作 与 开发 的 主流 技术 。 需 要 强调 的 是 ,在 学 习 JSP 技术 之 前 ,读者 
应 具有 较 好 的 Java 语言 基础 以 及 HTML 语言 基础 。 

2. JSP 引擎 

运行 包含 JSP 页 面 的 Web 项 目 , 需 要 一 个 支持 JSP 页 面 运 行 的 Web 服务 软件 ,该 服务 
软件 也 被 称 作 JSP 引擎 或 JSP 容器 。 通 常 将 安装 了 JSP 引擎 的 计算 机 称 为 一 个 支持 JSP 
的 Web 服务 器 。 当 用 户 要 访问 JSP 页 面 时 , Web 服务 器 通知 JSP 引擎 处 理 该 页 面 中 的 元 
素 ,返回 用 户 想 要 的 结果 ,如 图 2-1 所 示 。 

目前 ,比较 常用 的 JSP 引擎 有 Tomcat,JRun,Resin 等 ,本 书 采 用 的 是 Tomcat。 需 要 指 
出 的 是 ,JSP 引擎 需要 Java 语言 的 核心 库 和 相应 编译 器 ,所 以 在 安装 JSP 引擎 之 前 ,需要 先 
安装 J2SE(Java2 Standard Edition.Java2 标准 版 ) 提 供 的 JDK (Java Development Kit, Java 
开发 工具 包 ) , 详 见 2.1.2 节 。 

3. Web 服务 目录 

为 了 让 用 户 能 够 通过 浏览 器 访问 到 Tomcat 服务 器 上 的 JSP 页 面 ,就 必须 将 写 好 的 
JSP 页 面 保存 到 该 Tomcat 服务 器 的 某 个 Web 服务 目录 中 ,具体 包括 如 下 几 种 情况 。 
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Web 服 务 器 


JSP 引 擎 


[a ERER EREN 执行 


[s^] 

返回 结 执行 ”|JSP 页 面 对 应 的 
[Ek gw 
客户 


mw 客户 线程 字 节 码 


s 返回 结果 CT ic 


图 2-1 JSP 引擎 


1) 根 目录 

如 果 Tomcat 的 安装 目录 是 D:\apache-tomcat-6. 0.0, 那 么 Tomcat 服务 器 的 Web JR 
务 目录 的 根 目 录 就 是 D:\apache-tomcat-6. 0. 0\webapps\ROOT。 

用 户 如 果 要 访问 Web 服务 目录 根 目 录 中 的 JSP 页面, 在 浏览 器 地 址 栏 中 依次 输入 
Tomcat 服务 器 的 IP 地 址 、 端 口号 JSP 页 面 的 文件 名 即 可 。 例 如 ,Tomcat 服务 器 的 IP 地 
址 是 192. 168. 3. 1, 根 目录 中 存放 的 JSP 页 面 文件 名 为 first. jsp ,那么 用 户 在 浏览 器 地 址 栏 
需要 输入 http://192. 168. 3. 1:8080/first. jsp. 

2) 已 有 的 Web 服务 目录 

Tomcat 服务 器 安装 目录 中 的 webapps 目录 下 的 任何 一 个 子 目录 都 可 以 作为 Web 服 
务 目 录 。 安 装 Tomcat 后 ,webapps 目录 下 已 有 docs, examples, host-manager, manager 子 
目录 ,它们 都 可 以 作为 Web 服务 目录 ,也 可 以 在 webapps 目录 下 新 建 子 目 录 , 例 如 新 建 
myJSP 子 目录 ,那么 myJSP 就 成 为 一 个 Web 服务 目录 。 

如 果 将 JSP 页 面 保存 到 已 有 的 Web 服务 目录 中 ,那么 用 户 访问 页 面 时 需要 在 浏览 器 地 
址 栏 中 依次 输入 Tomcat 服务 器 的 IP 地 址 、 端 口号 、Web 服务 目录 名 、JSP 页 面 的 文件 名 。 
例如 ,Tomcat 服务 器 的 IP 地 址 是 192. 168. 3. 1,JSP 页 面 first. jsp 保存 在 webapps 目录 下 
的 examples 文件 夹 中 ,那么 用 户 在 访问 该 页 面 时 就 需要 在 浏览 器 地 址 栏 中 输入 http:// 
192. 168. 3. 1:8080/examples/first. jsp. 

3) 设置 新 的 Web 服务 目录 

可 以 将 计算 机 中 的 某 个 目录 设置 成 一 个 Web 服务 目录 ,并 为 该 Web 服务 目录 指定 虚 
拟 目录 ,也 就 是 说 ,隐藏 Web 服务 目录 的 真实 位 置 , 而 让 用 户 通过 虚拟 目录 访问 Web 服务 
目录 中 的 JSP 页面。 具体 操作 是 通过 修改 Tomcat 服务 器 安装 目录 下 的 conf 文件 夹 中 的 
server. xml 配置 文件 来 完成 。 例 如 要 将 D:\JSP fll E: NmyPages 设置 为 新 的 Web 服务 目 
录 , 让 用 户 通过 welcome 和 hello 虚拟 目录 来 访问 D:\JSP 和 E:\myPages 目录 中 的 JSP 页 
面 ,就 需要 查找 server. xml 配置 文件 中 的 一 /Host 二 部 分 (在 server. xml 文件 的 结尾 部 
分 ) ,在 其 前 面 加 入 代码 (代码 严格 区 分 大 小 写 ) : 


— Context path="/welcome" docBase— "D:NJSP" debug= "0" reloadable= "true" /> 
—Context path— "/hello" docBase="E:\myPages" debug= "0" reloadable= "true" /> 
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配置 文件 修改 后 ,需要 重新 启动 Tomcat 服务 器 才能 使 设置 生效 。 在 这 种 情况 下 访问 
JSP 页 面 时 ,用 户 需要 在 浏览 器 地 址 栏 中 依次 输入 Tomcat 服务 器 的 IP 地 址 、 端 口号 .虚拟 
目录 名 、JSP 页 面 的 文件 名 。 例 如 Tomcat 服务 器 的 IP 地 址 是 192. 168. 3. 1 ,用户 要 访问 
D:\JSP 或 E:\myPages 中 的 JSP 页 面 first. jsp, 则 需要 在 浏览 器 地 址 栏 中 输入 : 


http:// 192.168.3.1:8080/welcome/first.jsp 或 http:// 192.168.3.1:8080/hello/first.jsp 


2.1.2 技能 操作 


搭建 JSP 开发 平台 ,具体 步骤 如 下 : 

。 安装 与 配置 JDK。 

。 安装 与 配置 Tomcat。 

* 编写 .保存 .运行 JSP 页 面 。 

1. 安装 与 配置 JDK 

D FR ZX JDK F 

(1) 登录 http://www. oracle. com/technetwork/java, 在 — 
Software Downloads 里 选择 Java SE 提供 的 JDK ,结合 本 地 计算 
机 所 使 用 的 操作 系统 和 具体 的 机 器 位 数 ,在 下 载 列表 中 选择 相应 。 ja vindowsi 
W JDK 下 载 到 本 地 计算 机 的 指定 目录 。 

(2) 完成 下 载 后 ,在 目录 中 会 出 现 如 图 2-2 所 示 的 安装 程序 


图 标 ,双击 该 图 标 进行 JDK 的 安装 。 
(3) 在 弹出 的 许可 证 协议 对 话 框 中 单 击 “ 接 受 " 按 钮 ,如 图 2-3 所 示 。 


[$ Java(TM) SE Development Kit 6 -许可 还 =) 


许可 证 协议 ` 
WPA TEENE o È Sun. 


2-2 JDK 安装 程序 


Sun Microsystems, Inc. Binary Code License Agreement 


for the JAVA SE DEVELOPMENT KIT (JDK), VERSION 6 


SUN MICROSYSTEMS, INC. ("SUN") IS WILLING TO LICENSE THE 

SOFTWARE IDENTIFIED BELOW TO YOU ONLY UPON THE CONDITION 

[THAT YOU ACCEPT ALL OF THE TERMS CONTAINED IN THIS 

BINARY CODE LICENSE AGREEMENT AND SUPPLEMENTAL LICENSE 

[TERMS (COLLECTIVELY "AGREEMENI").U PLEASE READ THE 
CAREFULLY.D BY DOWNLOADING OR INSTALLING THIS | 


|| installshield 


接受 A) > 


2-3 ”接受 许可 协议 
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CD 在 弹出 的 自 定 义 安 装 对 话 框 中 ,默认 的 安装 路 径 为 C:\Program Files\Java\jdk1 
.6.0\, 单 击 “ 更 改 ” 按 钮 可 以 更 改 安装 路 径 , 如 图 2-4 所 示 。 


SE Development Kit 6 - EX — 
自 定义 安装 
| 选择 要 安装 的 程序 功能 。 


从 中 选择 ° 安装 完成 后 ， 您 可 以 使 用 "控制 面板 "中 的 "添加 / 
| 有 li 
UP 
SE Development Kit 6 
ust p uer JRE 6o i 
e- ET ahi BC ERA EI 
Lan D] As RE 
安装 到 : 
C:\Program Files\Java\jdk1.6.0\ ERA 
InstallShield 


图 2-4 自 定义 安装 
(5) 单 击 “下 一 步 ? 按 钮 开始 安装 ,如 图 2-5 所 示 。 


正在 安装 您 选择 的 程序 功能 。 @Sun 


B ZADE Java(TM SE Development Kit 6, 请 稍 假 。 这 可 能 需 
deb: 


— —— —nmm—Á À | 


图 2-5 安装 进度 


(6) 进度 完成 后 ,弹出 自 定义 安装 JRE(Java Runtime Environment. Java 运行 时 环境 ) 
对 话 框 ,默认 的 安装 路 径 为 C:\Program Files\Java\jrel. 6.0\, 单 击 “ 更 改 ” 按 钮 可 以 更 改 安 
装 路 径 , 如 图 2-6 所 示 。 
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安装 完成 


7 Java(TM) SE Development Kit 6° 单 


显示 自述 文件 


图 2-7 安装 完成 


2) 配置 环境 变量 Java_home, Path 和 Classpath 

CD 右 击 “我 的 电脑 "图标, 依次 单 击 “ 属 性 ”一 “高 级 系统 设置 ”>“ 高 级 ”>“ 环 境 变量 ” 
按钮 ,如 图 2-8 所 示 。 

(2) 单 击 “ 环 境 变 量 ” 按 钮 后 ,弹出 如 图 2-9 所 示 的 “环境 变量 ”对 话 框 ,在 这 个 对 话 框 中 
可 以 分 别 设置 环境 变量 Path、Classpath、Java_home 的 值 。 
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要 进行 大 多 数 更 改 ， 您 必须 作为 管理 员 登 录 。 
性 能 
视觉 效果 ， 处 理 器 计划 ， 内 存 使 用 ， 以 及 虚拟 内 存 


用 户 配置 文件 
与 您 登录 有 关 的 点 面 设置 


(800... ) [iR D...) 


值 
ConSpec C; MindowsVsysten32 Vend. exe 
FPONO HDST C... NO 
NUMBER OF PR... 2 
ns 


Windows NT 


(ss an...) 


29 环境 变量 


在 设置 前 ,首先 检查 “用 户 变量 "和 * 系 统 变量 ”中 是 否 已 经 有 Path 环境 变量 存在 ,如 果 
没有 可 单 击 “ 系 统 变量 ?选项 组 中 的 “新 建 ? 按 钮 ,弹出 如 图 2-10 所 示 的 “新 建 系统 变量 ”对 话 
框 ,在 该 对 话 框 的 “变量 名 ”文本 框 中 输入 Path, 在 “变量 值 ”文本 框 中 输入 C:\Program 
Files\Java\jdk1. 6. 0\bin 即 可 完成 Path 环境 变量 的 设置 ; 如 果 事 先 检 查 到 已 经 有 Path 环 
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境 变 量 存 在 , 则 首先 选中 Path, 然 后 单 击 “ 编 辑 ” 按 钮 ,弹出 如 图 2-11 所 示 的 “编辑 系统 变量 ”对 
话 框 。 在 “变量 值 ” 文 本 框 中 编辑 Path 的 值 ,将 需要 的 值 C:\Program Files\Java\jdk1. 6. ON 
bin 加 入 即 可 ,但 需要 注意 的 是 ,新 加 入 的 值 最 好 放 在 最 前 面 , 且 与 原 有 的 值 用 分 号 间隔 开 ， 
以 保证 Java 程序 和 JSP 页 面 的 正常 运行 。 


rara sassa =] 
一 - — - 
mE Path 变星 名 中 Path 
变量 值 D >:AProgran Files\Java\jdkl. 6.0\bin 变量 值 OD: 2:\Progran Files\Java\jdkl. 6.0\bin;] 
C J Com) 
图 2-10 新 建 系统 变量 图 2-11 编辑 Path 


类 似 地 , 将 Classpath 的 值 设 置 为 “C:\Program Files\Java\jdkl. 6. ONjreNlibN rt 
„jars. ;” 即 可 (其 中 的 “. "表示 当前 路 径 ) ,如 图 2-12 所 示 。 

最 后 ,将 Java home 的 值 设 置 为 C:\Program Files\Java\jdk1. 6.0, 如 图 2-13 所 示 。 
mass gga 


EEE Java hone 
KEBO C:\Progran Files Java jaki 6.0] 


Coe J[ má J 


Classpath 


les V Java jdkl 6. O\jre\lib\rt. jar;. ; 


Ce J[ má J 


图 2-12 iE Classpath 图 2-13 iE Java home 


3) 测试 JDK 

CD 打开 一 个 文本 编辑 器 (如 记事 本 程序 ) ,按照 下 面 “例子 程序 代码 ”的 内 容 输 入 源 程 序 。 
例子 程序 代码 : 

public class Test( 


public static void main(String args 口 ){ 
System. out. println(" 测 试 JDK ,安装 成 功 "); 


|| 
|| 
(2) 将 上 述 源 程序 保存 到 D 盘 的 某 个 文件 夹 中 ,例如 c: uers naninistrator>a: 
D:\java。 要 求 保存 类 型 为 “所 有 文件 ”. 文 件 名 为 Test | MOTERA 
. Java, D: Njava»javac Test.java 


(3) 在 计算 机 中 选择 “开始 ”所 有 程序 "一 "附件 ”一 |， e 
“命令 提示 符 ”命令 ,运用 javac 命令 编译 源 文件 ,运用 java W = 
命令 运行 程序 ,如 图 2-14 所 示 。 图 2-14 测试 JDK 
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2. 安装 与 配置 Tomcat 
1) 安装 Tomcat 服务 器 


安装 Tomcat 之 前 应 确保 事先 已 成 功 安装 TDK ,然后 登 Wi REET 
3K http: //tomcat. apache. org/ 下 载 Tomcat 程序 ,然后 将 bin 
其 解压 到 任意 盘 符 下 ,这 里 将 其 解压 到 D:\, 解 压 后 将 出 现 
如 图 2-15 所 示 的 目录 树 结构 。 logs 

2) 启动 Tomcat 服务 器 mp 

进入 Tomcat 服务 器 的 安装 目录 ,找到 bin 目录 中 的 uuu 


startup . bat 批 处 理 文件 ， 然后 双 击 ， 即 可 启动 服务 器 。 执 行 

startup. bat 启动 Tomcat 服务 器 后 ,会 占用 一 个 MS-DOS 窗口 ， 图 2-15 Tomcat 服务 器 目录 
如 图 2-16 所 示 , 如 果 关 闭 当 前 MS-DOS 窗口 也 就 意味 着 将 关 树 结构 

闭 Tomcat 服务 器 。 


32 org.apache.catalina.core.fprLifecycleListener lifecycleEvent 
The Apache Tomcat Native library which allow 
s was not found on the jav: 
Windows \Sun WJava\bin 


32 org.apache.coyote.httpii.HttpiiProtocol init 
ing Coyote HTTP/1.1 on http-8880 

32 org.apache.catalina.startup.Catalina load 
ation processed in 1529 

:33 org.apache.catalina.core.StandardService start 
service Catalina 

33 org.apache.catalina.core.StandardEngine start 


Servlet Engine: hpache Toncat/6.0.8 
33 org.apache.catalina.core.StandardHost start 
XML validation disabled 
13 9:18:34 org -apache -coyote.httpll.HttpllProtocol start 
Starting Coyote HITP/1.1 on http-8989 
8:34 org.apache.jk.common.ChannelSocket init 
ajp13 listening on /0.0.0.0:8009 
8:34 org.apache.jk.server.JkMain start 
running ID-8 tine-8/47 config-null 
34 org.apache.catalina.startup.Catalina start 


图 2-16 启动 Tomcat 


/127. 0. 0. 


Tomcat 启动 后 ,在 浏览 器 的 地 址 栏 中 输入 http: //localhost: 8080 ak http: 
1:8080 ,会 出 现 如 图 2-17 所 示 的 测试 页 面 。 需 要 说 明 的 是 ,Tomcat 服务 器 占用 的 默认 端口 
号 是 8080, 如 果 Tomcat 所 使 用 的 端口 已 经 被 占用 将 会 造成 Tomcat 无 法 正常 启动 。 当 然 ， 
也 可 以 通过 修改 Tomcat 服务 器 conf 文件 下 的 主 配置 文件 sever. xml 来 指定 Tomcat Bf (H 
用 的 端口 号 。 用 记事 本 打开 server. xml 文件 ,查找 如 下 代码 : 


所 Connector port= "8080" protocol="HTTP/1.1" 
maxThreads= "150" connectionTimeout= "20000" 
redirectPort= "8443" /> 
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将 其 中 的 port=8080 更 改 为 新 的 端口 号 并 重新 启动 Tomcat 服务 器 ,例如 可 将 端口 号 改 为 
8888 等 。Tomcat 安装 时 默认 的 端口 设置 的 是 8080 ,而 http 协议 的 默认 端口 是 80 ,所 以 测 
iX Tomcat 时 需要 在 浏览 器 的 地 址 栏 输入 http://localhost: 8080 或 http://127. 0. 0. 1: 
8080 , 若 把 Tomcat 的 端口 设置 为 80, 则 直接 输入 http://localhost 或 http://127.0.0.1 就 
能 显示 Tomcat 测试 页 面 。 


httpi/localhost8080, $ O ~ Ó | [a] Apache Tomcat x 
x» EE punqa BIBOMRRUS Y 


Apache ^I 
Toma The Apache Software Foundation ` 
http://www.apache.org/ 


If you're seeing this page via a web browser, it means you've | 
setup Tomcat successfully. Congratulations! 


As you may have guessed by now, this is the default Tomcat home “| 
page. It can be found on the local filesystem at: 


SCATALINA HOME/webapps/ROOT/index.html 


where "SCATALINA HOME" is the root of the Tomcat installation 
directory. If you're seeing this page, and you don't think you should 
be, then either you're either a user who has arrived at new 
installation of Tomcat, or you're an administrator who hasn't got 
his/her setup quite right. Providing the latter is the case, please refer 
to the Tomcat Documentation for more detailed setup and 


l= administration information than is found in the INSTALL file. 
2-17 Tomcat 测试 页 面 


v 


3) 关闭 Tomcat 服务 器 

前 面 已 经 提 到 ,直接 关闭 Tomcat 所 占用 的 MS-DOS 窗口 就 意味 着 关闭 Tomcat 服务 
器 。 除 此 之 外 ,还 有 两 种 关闭 方法 : 其 一 是 进入 Tomcat 服务 器 的 安装 目录 ,找到 bin 目录 
中 的 shutdown. bat 批 处 理 文件 ,双击 即 可 关闭 服务 器 ; 其 二 是 直接 使 用 快捷 键 Ctrl 十 C 关 
B] Tomcat 服务 器 。 

3. 编写 .保存 .运行 JSP 页 面 

1) 编写 JSP 代码 

打开 一 个 文本 编辑 器 。 如 果 是 Windows 操作 系统 ,要 打开 “记事 本 ”编辑 器 ,可 以 通过 
“开始 ”一 "程序 ”附件 ”一 记事 本 ”命令 来 完成 ; 如 果 是 其 他 操作 系统 ,请 参考 操作 系统 
的 帮助 手册 打开 一 个 文本 编辑 器 。 在 打开 的 文本 编辑 器 中 输入 下 面 的 代码 : 

<%@ page contentType= "text/html;charset— GB2312" %> 


<html> 
<head> 
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«title JSP Ji nj </title> 
</head> 
<body bgcolor= "gray" > 
<% 
out. print(" 这 是 我 的 第 一 个 JSP 页 面 "); 
%> 
<h1>if#8 1 到 100 2 f — /h17 
Aint i, sum—0; 
for(i=1;i<=100;i++) 
sum+ =i; 
x» 
结果 : < ⁄4=sum⁄> 
</body> 
</html> 


2) 保存 JSP 页 面 

将 编辑 的 JSP 代码 保存 到 Tomcat 服务 器 的 Web 服务 目录 ,此 处 选择 保存 到 Web 服 
务 目录 的 根 目 录 中 , 即 保存 到 D: Napache-tomcat-6. 0. 0\webapps\ROOT。 文 件 名 为 first 
.jsp, 其 中 .jsp 是 文件 的 扩展 名 。 使 用 记事 本 编辑 JSP 文件 ,在 保存 时 ,必须 将 “保存 类 型 ” 
选择 为 “所 有 文件 ”, 将 “编码 "选择 为 ANSI, 如 图 2-18 所 示 。 


文件 名 (N): firstjsp ` 
保存 类 型 中 : | 所 有 文件 M 
pe sp — 0 7 MA 


2-8 保存 JSP 页 面 


3) 运行 JSP 页 面 

因为 在 保存 JSP 页 面 时 ,直接 保存 到 了 Web 服务 目录 的 根 目录 中 , 即 保存 到 D: N 
apache-tomcat-6. 0. 0\webapps\ ROOT ,所 以 在 运行 JSP 页 面 时 直接 在 浏览 器 地 址 栏 中 输 
入 http://localhost:8080/first. jsp 即 可 查看 页 面 效果 ,如 图 2-19 所 示 。 


r 


-— AN 


€ SE httpy/localhost8080/firstjsp 8 O - Ó | [ad 


图 2-19 3247 JSP HH 
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2.1.3 拓展 训练 


1. 编写 一 个 Java 应 用 程序 ,要 求 能 够 输出 100 以 内 的 奇数 。 

2. 修改 Tomcat 服务 器 的 主 配置 文件 server. xml, 将 默认 端口 号 8080 修改 为 9090, 然 
后 测试 Tomcat 服务 器 是 否 能 够 正常 启动 ,打开 浏览 器 在 地 址 栏 中 输入 相应 的 地 址 ,查看 能 
否 出 现 测试 页 面 。 成 功 后 ,将 端口 号 改 回 8080, 重 新 测试 Tomcat 服务 器 是 否 能 够 正常 启 
动 ,打开 浏览 器 在 地 址 栏 中 输入 相应 的 地 址 ,再 次 查看 能 否 出 现 测试 页 面 。 

3. 如 果 E:\myJSP 是 一 个 Web 服务 目录 ,其 虚拟 目录 为 game, index. jsp 保存 在 E:\ 
myJSP\ch02 中 ,那么 在 Tomcat 服务 器 (端口 号 为 8080) 所 在 计算 机 的 浏览 器 地 址 栏 中 输 
入 下 列 哪个 地 址 能 够 正确 访问 到 index. jsp 页面 7(  ) 

A. http://localhost:8080/index. jsp 

B. http://localhost :8080/myJSP/ch02/index. jsp 
C. http://localhost:8080/game/index. jsp 

D. http: //localhost:8080/game/ch02/index. jsp 

4. 请 在 D:\ 盘 建立 一 个 名 为 study 的 文件 夹 , 并 将 该 目录 设置 为 一 个 Web 服务 目录 ， 
然后 用 文 编辑 器 编写 一 个 简单 的 JSP 页面, 保存 到 该 目录 中 ,让 用 户 通过 虚拟 目录 find 来 
访问 这 个 JSP 页 面 。 


2.2 使 用 IDE 工具 开发 JSP AH 


2.2.1 知识 要 点 


1. JSP 页 面 

JSP 页 面 中 可 以 有 普通 的 HTML 标记 ,也 可 以 有 规定 的 JSP 标记 ,以 及 通过 标记 符 
"S TUAE Java 程序 片段 。 在 保存 JSP 页 面 时 ,文件 名 必须 符合 标识 符 的 规则 , 即 
由 字母 .数字 、 下 划 线 和 美元 符号 组 成 , 且 不 能 以 数字 字符 作为 开头 。 一 个 JSP 页 面 文件 的 
扩展 名 为 . jsp。 

2. IDE 工具 

2.1 节 介绍 了 使 用 文本 编辑 器 即 记 事 本 程序 编写 JSP 文件 的 方法 ,除了 文本 编辑 器 之 
外 ,还 可 以 使 用 IDE(Integrated Development Environment, 集 成 开发 环境 ) 工 具 来 帮助 程序 
员 快 速 开发 JSP 程序 ,常用 的 IDE 工具 有 Eclipse、NetBeans、JBuilder 等 ,本 书 采用 Eclipse。 
Eclipse 是 由 IBM 等 多 家 公司 和 组 织 成 立 的 Eclipse 基金 会 开发 的 开放 源 代码 工具 , 它 本 身 
是 一 个 框架 和 一 组 服务 ,通过 插件 机 制 来 灵活 地 构建 开发 环境 。 


2.2.2 技能 操作 
使 用 Eclipse 集成 开发 环境 创建 Web 项 目 , 在 项 目 中 创建 JSP 文件 ,然后 部 署 项 目 到 
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Tomcat 服务 器 并 运行 。 具 体 步 骤 包 括 : 

(1) 安装 启动 Eclipse, 

(2) 配置 服务 器 。 

(3) 创建 Web m AA JSP 文件 。 

(4) 编写 .调试 .运行 JSP 页 面 。 

1. 安装 启动 Eclipse 

登录 官方 网 站 http://www. eclipse. org 下 载 相 应 版 本 的 Eclipse SDK ,通常 为 压缩 包 
格式 ,解压 后 双击 eclipse 应 用 程序 图 标 便 可 启动 进行 使 用 。 如 果 下 载 的 版 本 为 安装 版 , 则 
需要 遵照 安装 向 导 指 引进 行 安装 后 方 可 启动 进行 使 用 。 

启动 时 会 弹出 如 图 2-20 所 示 的 对 话 框 ,用 于 选择 工作 区 , 即 选 择 Web 项 目 存 放 的 目录 
位 置 。 


Select a workspace 
Eclipse stores your projects in a folder called a workspace. 


Use this as the default and do not ask again 


2-20 选择 工作 区 


图 中 上 默认 目录 位 置 为 C:\Users\Administrator\workspace, 如 果 想 对 此 进行 更 改 , 可 单 
击 Browse 按钮 进行 设置 。 选 择 工 作 区 之 后 , 单 击 右 下 角 的 OK 按钮 , 便 可 成 功 启动 
Eclipse, 启 动 后 的 界面 如 图 2-21 所 示 。 

从 图 中 可 以 看 出 ,Eclipse 界面 的 顶端 与 很 多 应 用 软件 类 似 , 依 次 是 标题 栏 .菜单 栏 . 工 
具 栏 ,标题 栏 显示 的 是 软件 的 标题 信息 ,菜单 栏 中 的 菜单 和 工具 栏 中 的 工具 能 够 帮助 开发 人 
员 快 速 完 成 所 需要 的 操作 。 界 面 的 左 侧 是 工程 浏览 器 ,展示 了 在 此 建立 的 Java 项 目 或 Java 
Web 项 目的 树 型 结构 目录 ,图 2-21 中 只 有 一 个 Java Web 项 目 名 为 HelloWorld。 界 面 中 间 
较 大 的 区 域 是 编辑 区 ,在 编辑 区 开发 人 员 可 以 进行 代码 的 编写 ,一 般 默 认 情 况 下 Eclipse 会 
有 一 些 自 带 的 生成 代码 ,实际 开发 时 可 以 根据 需要 选择 是 否 留用 或 全 部 删除 后 重新 编写 。 
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- [LN dr an = 232 > E [3 


(bPrjeatxp.:3 = O [Q SPRM |Ð Newiespz| 1 so ar = n 
Bgl 了 1 kx@ page language="java" contentType-"text/h a 
S HelloWorld 2 pagetncoding-"I50-8859-1"X» 
4 W HelloWor 3 <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 
b AP JAX-WS Web Service | — 45 «html» 
y loyment Descr 58 «head» 
Tad P| 6 <meta http-equiv=“Content-Type" content-"tex 
> WP Java Resources 7 «title»Insert title herec/title» 
b BÀ JavaScript Resource 8 «/head» 
» © build 
» > WebContent 
b (@ Servers 12 </html> 


| | 
图 Markers [D] Proper... Mb Servers 53 W Data So... [5 Snippets © Console ° O 
BtomEr 


B, Tomcat v6.0 Server at localhost [Started, Restart] 


| Writable | SmartInsert |1:1 


2-21 Eclipse 界面 


界面 的 右 侧 是 工程 概要 区 ,这 里 可 以 显示 代码 中 的 变量 和 方法 ,程序 员 能 够 借 此 快速 定位 代 
码 中 变量 和 方法 ,便于 修改 和 调试 。 界 面 的 底 端 是 日 志 查 看 区 ,用 于 追踪 软件 运行 ,查看 错 
误 日 志 。 

2. 配置 服务 器 

Eclipse 安装 完成 后 ,还 需要 为 其 配置 Web 服务 器 ,具体 操作 步骤 如 下 : 

(1) 在 Eclipse 窗口 中 逐 级 选择 Window — Preferences — Server 一 Runtime 
Environment 选项 ,如 图 2-22 所 示 , 

(2) 因为 此 前 从 未 进行 过 服务 器 的 相关 设置 ,所 以 需要 单 击 右 侧 的 Add 按钮 ,来 添加 
一 个 新 的 服务 器 运行 时 环境 。 前 面 介绍 过 ,我们 已 经 安装 了 Tomcat 6.0 作为 Web 服务 器 ， 
所 以 在 此 选择 Apache Tomcat v6.0, 如 图 2-23 所 示 。 

(3) 单 击 Next 按钮 ,在 弹出 的 对 话 框 中 单 击 Browse 按钮 ,可 以 在 接 下 来 弹出 的 “浏览 
文件 夹 ” 对 话 框 中 选择 Tomcat 的 安装 目录 ,在 此 选择 的 是 D:\apache-tomcat-6. 0. 0, 如 
图 2-24 所 示 。 

(4) Tomcat 安装 目录 选 好 后 单 击 “ 确 定 ” 按 钮 即 可 。 在 该 对 话 框 中 继续 单 击 Install 
JREs( 安 装 Java 运行 时 环境 等 ) 按 钮 ,会 弹出 如 图 2-25 所 示 的 对 话 框 ,在 该 对 话 框 中 进行 后 
续 JRE 的 设置 。 
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图 2-22 设置 服务 器 运行 时 环境 


New Server Runtime Environment 
Define a new server runtime environment 


Select the type of runtime environment: 
type filter text 


Le 会 ObiectWeb ad 
Apache Tomcat v6.0 supports J2EE 1.2, 1.3, 14, and Java EE 5 and 6 Web modules. 


E Create a new local server 


2 <Back [Nessa] Eu] [ces] 


2-23 ”添加 新 的 服务 器 运行 时 环境 
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Specify the installation directory 


Ji. 27.0.1430.0_chrome_installer 
> JÈ 360Downloads 
》 201506010615 关于 创业 2015 年 省 社 和 
> À: apache-tomcat-5.5.20 
^ pudtonccóto 
"— | 


MRF): spachectoncat-8.0.0 


Add, remove or edit JRE definitions. By default, the checked JRE is added to the build path of newly 
created Java projects. 

Installed JREs: 

Name Location Type 

| májdkL60.. CWProgram Files\Java\jdk1.6.0 Standard V... 


2-25 JRE 设 置 
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(5) 如 果 这 里 已 经 配置 了 JDK, 可 以 在 此 直接 进行 选择 ,同时 也 可 以 单 击 右 侧 的 Edit. 
Duplicate, Remove, Search 按钮 分 别 进行 编辑 、 复 制 . 移 除 、 搜 索 等 相关 操作 。 如 果 之 前 没有 
配置 过 JDK, 则 需 单 击 右 侧 的 Add 按钮 ,新 增 一 个 标准 的 JRE, 如 图 2-26 所 示 。 


(8) Add JRE 


JREType 
Select the type of JRE to add to the workspace. 


Installed JRE Types: 


Execution Environment Description 
Standard 1.1.x VM 


Standard VM 


|o [ -— e c imas) 


2-26 ”新 增 标准 的 JRE 


(6) 在 如 图 2-26 所 示 的 添加 JRE 对 话 框 中 单 击 Next 按钮 后 ,弹出 如 图 2-27 所 示 的 
JRE 安装 目录 选择 界面 , 单 击 该 界面 中 JRE home 文本 框 后面 对 应 的 Directory 按钮 ,选择 
好 目录 然后 单 击 “ 确 定 ” 按 钮 ,在 此 选择 的 是 C:\Program Files\Java\jdk1. 6. 0。 

经 过 上 述 步 又, Web 服务 器 的 配置 已 经 完成 。 


[8] Add JRE = 
— =m“ 
JRE Definition 
Â Enter the home directory of the JRE. i 
me [moo 
JRE name: 
Select the root directory of the JRE 
installation: [ade Enternal MM. 
Javadoc Location... 
b À Internet Explorer. === 
aB Jam Source Attachment... 
aldi idkt60 Remove 


Ë bin 


2-27 选择 JRE 安装 目录 
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3. 创建 web 项 目 和 JSP 文件 
(1) 进行 完 服务 器 的 配置 , 接 下 来 就 可 以 创建 Java Web 项 目 了 。 在 Eclipse 主 窗口 逐 
级 选择 File New- Dynamic Web Project 菜单 命令 来 创建 项 目 , 如 图 2-28 所 示 。 


Al+Shif+N » | Ë JPA Project 


2-28 创建 Web 项 目 


(2) 在 弹出 的 对 话 框 中 ,可 以 进行 Web 项 目的 详细 设置 ,如 在 Project name 文本 框 中 
为 项 目 进行 命名 ,本 例 中 命名 为 HelloWorld; 同时 在 Target runtime 下 拉 列 表 中 选择 之 前 
配置 过 的 服务 器 Apache Tomcat v6.0, 如 图 2-29 所 示 。 


Create a standalone Dynamic Web project or add it to a new or existing Enterprise Application. 


Dynamic Web Project - | 


Project name: HelloWorld 
Project location 
Use default location 


Location: | CAUsers Administrator workspaceWHelloWorld J|. Browse.. 
pe | 
New Runtime... 

| «None» 

a) = 
Configuration 

Default Configuration 7| | Modify... 

The default configuration provides a good starting point. Additional facets can later be installed to add || 
new functionality to the project. 
EAR membership 

Add project to an EAR 

EAR project name: | EAR ~ | | New Project... | 


Working sets 

[E] Add project to working sets 

Working sets: ` Select. | 
Q [ «Bak [Nene Finish Cancel 


图 2-29 Web 项 目的 详细 设置 
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(3) 单 击 Next 按钮 后 ,进入 下 一 步 , 按 照 向 导 提示 进行 完 所 有 的 设置 后 单 击 Finish 按 
钮 完成 Web 项 目的 创建 。 项 目 创建 后 ,在 Eclipse 窗口 左 侧目 录 树 中 就 会 出 现 对 应 的 目录 
结构 ,在 该 结构 中 的 WebContent 节点 上 右 击 ,依次 选择 New—JSP File 命令 ,新 建 一 个 JSP 
文件 ,如 图 2-30 所 示 。 新 建 的 JSP 文件 ,默认 名 称 为 NewFile. jsp, 可 以 根据 实际 需求 对 该 
文件 名 进行 修改 。 


a HelloWorld 
b 4) MOCWS Well — New = 
b $ Deployment Go Into 
b Ë Java Resour 
b mÀ JavaScript Re Show In AlteShifteW > 
» 包 buid | 加 copy C 
» (E WebContent a Copy Qualified Name 

4 E Test QD Peste ANN 
b $ JAX-WS Well — 
b Ña Deployment| 86 | Delete Delete 
b 89 Java Resour| — Remove from Context Ctrl +Alt+Shift+Down 
b mÀ JavaScript Ri Build Path Ç: 


图 2-30 创建 JSP 文 件 


4. 编写 .调试 .运行 JSP 页 面 

1) 编写 代码 

在 新 建 的 JSP 文件 中 ,会 自动 生成 部 分 代码 ,开发 人 员 只 需要 在 核心 部 分 编写 关键 代 
码 即 可 。 下 面 是 一 个 简单 的 JSP 文件 ,文件 名 为 NewFile. jsp. HM Hf < body > 55 
三 /body 二 标记 间 的 内 容 为 开发 人 员 手 动 输入 ,其 余 代 码 均 为 自动 生成 。 


— (8 page language= "java" contentType= "text/html; charset— ISO-8859-1" 
pageEncoding — "ISO-8859-1" > 

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" 

"http://www. w3.org/ TR/html4/loose. dtd" > 

<html> 

<head> 

<meta http-equiv= "Content-Type" content=" text/html; charset— ISO-8859-1"> 
<title> Insert title here</title> 

</head> 

<body> 

HelloWorld! 

This is my first JSP file! 

</body> 

</html> 


2) 调试 .运行 JSP 页 面 

编写 好 的 代码 如 果 有 语法 错误 ,需要 手动 修改 。 如 果 没 有 错误 , 则 在 Eclipse 左 侧目 录 
结构 中 找到 该 文件 ,然后 右 击 , 依 次 选择 Run As 一 Run on Server 菜单 命令 ,如 图 2-31 所 
示 。 在 打开 的 Run On Server 对 话 框 中 ,选中 Always use this server when running this 
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project 复 选 框 , 其 他 选用 默认 设置 ,如 图 2-32 所 示 。 


: e È Paste Ctrl+V 
b @ META-INF | SE | Daine Delete 
b @> WEB-INF 
(Ex Nestle] > Remove from Context Ctrl Alt«Shift Down. 
4] Mark as Landmark Ctr Alt«Shift«Up. 
Build Path , 


Validate 
Show in Remote Systems view 
Profile As 

Debug As 

Run As 

Team 

Compare With 


2-31 运行 JSP 页 面 的 方法 


Run On Server | 
Select which server to use 


| How do you want to select the server? 
|| @ Choose an existing server 

© Manually define a new server 
Select the server that you want to use: 
type filter text o _ 


Server State 
4 @> localhost 
Tomcat v6.0 Server at localhost B, Started 


| Apache Tomcat v6.0 supports J2EE 1.2, 1.3, LA, and Java EE S and 6 Web — (Columns. 
|| modules. 


[V] Always use this server when running this project 


© 


图 2-32 Run on Server 对 话 框 
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前 面 提 到 的 NewFile. jsp 文件 代码 ,在 Eclipse 运行 后 的 效果 如 图 2-33 所 示 。 


一 = BH > http://localhost:8080/HelloWorld/NewFilejjsp 四 [^| 
Hello World! This is my first JSP file! 


2-33 ”查看 JSP 页 面 运行 效果 


需要 注意 的 是 ,如 果 在 浏览 器 中 运行 上 面 的 JSP 页 面 , 则 需要 在 浏览 器 地 址 栏 中 输入 
http://localhost: 8080/HelloWorld/NewFile. jsp. 也 就 是 说 , Eclipse 建立 的 项 目 
HelloWorld 会 被 认为 是 一 个 Web 服务 目录 。 


2.2.3 拓展 训练 


1. 参考 本 节 技 能 操作 中 创建 Web 项 目 和 JSP 文件 的 步骤 ,创建 一 个 名 为 Test 的 Java 
Web 项 目 , 并 在 该 项 目 中 创建 一 个 名 为 myTest. jsp 的 文件 。 

2. 参考 本 节 技 能 操作 中 编写 、 调 试 、 运 行 JSP 页 面 的 步骤 ,在 myTest. jsp 页 面 中 显示 
“这 是 我 创建 的 JSP 文件 ,全 都 学 会 了 ,好 高 兴 啊 !”, 并 将 其 发 布 运行 。 

3. 思考 如 何 将 已 有 代码 导入 到 Eclipse 中 ,并 试 着 完成 操作 。 


2.3 小 结 


JSP 技术 不 仅 是 开发 Java Web 的 主流 技术 ,而 且 也 是 进一步 学 习 相关 技术 (如 模式 
和 框架 ) 的 基础 。 

JSP 引擎 是 支持 JSP 程序 的 Web 服务 器 ,负责 运行 JSP 程序 ,并 将 相关 结果 发 送 给 
客户 端 。 目前 流行 的 JSP 引擎 有 Tomcat, Resin、JRun、WebSphere、WebLogic 等 ， 
本 书 使 用 的 是 Tomcat 服务 器 。 

当 服 务 器 上 的 一 个 JSP 页面 第 一 次 被 客户 请 求 执行 时 ,服务 器 上 的 JSP 引擎 首先 将 
JSP 文件 转译 成 一 个 Java 文件 ,并 将 Java 文件 编译 成 字 节 码 文件 ,然后 执行 字 节 码 
文件 响应 客户 端的 请 求 。 

* 安装 Tomcat 服务 器 ,首先 安装 JDK ,并 需要 设置 Java home 环境 变量 。 


JSP 基本 语法 


本 章 将 JSP 所 包含 的 各 类 元 素 ,如 声明 注释、 程序 片 . 表 达 式 JSP 指令 标记 (Directive 
Elements) JSP 动作 标记 (Action Elements) 等 主要 方面 来 说 明 JSP 的 基本 语法 结构 。 


3.1 JSP 文件 的 构成 


3.1.1 知识 要 点 


在 传统 的 网 页 即 HTML 文件 中 插入 Java 动态 元 素 、JSP 标记 等 ,就 形成 了 JSP 页 面 。 
通常 ,一 个 JSP 文件 包括 如 下 组 成 部 分 : 

(OD HTML 标记 ,这 在 第 1 章 中 已 有 介绍 。 

(2) Java 动态 元 素 ,包括 变量 和 方法 的 定义 、Java 程序 片 和 Java KER., 

(3) 注释 ,包括 HTML 注释 和 JSP 特有 的 注释 。HTML 注释 格式 为 : <! -- 注 释 内 
容 -- 二 ,JSP 特有 的 注释 格式 为 : 


二 %-- 注 释 内 容 --% 二 
(4) JSP 标记 ,包括 JSP 指令 标记 和 JSP 动作 标记 。 


3.1.2 技能 操作 


识别 JSP 文件 中 的 构成 要 素 , 具 体 任务 如 下 : 
。 阅读 下 面 的 代码 ,识别 其 中 的 JSP 构成 要 素 。 
。 分别 在 记事 本 和 Eclipse 中 输入 下 面 的 代码 ,并 对 JSP 页 面 进行 调试 运行 。 
1. 阅读 下 面 的 代码 ,识别 其 中 的 JSP 构成 要 素 。 
代码 模板 eg3_1. jsp MF: 
<%@ page language= "java" contentType= "text/html; charset— GBK" 
pageEncoding— "GBK" Yi >< !-- JSP 指令 标记 --> 
<%-- F Hl JE B (9 558 RU TIRE 30-706 
<%! 
int i=0; // 变 量 声明 
double mul(double a, double b){ // 方 法 定义 


return a * b; 
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) 
%> 
<html> 
<head> 
<title>JSP X fFF IH — / title 
</head> 
<body bgcolor= "cyan" >< !--HTML 标记 --> 
«Nico: 
double mulitiple? mul(5.5,6.0); //Java 程序 片 
HW 
第 
<%=i X> 
次 计算 ,结果 为 
<%= mulitiple 47 <!--Java 表达 式 --> 
</body> 
</html> 


2. 分 别 在 记事 本 和 Eclipse 中 输入 上 面 eg3_1. JSP 中 的 代码 ,并 对 JSP 页 面 进行 调试 
运行 。 

1) 使 用 记事 本 和 浏览 器 完成 

按照 2. 1 节 介绍 的 方法 ,打开 记事 本 编写 代码 ,然后 将 代码 保存 到 自己 设 定 的 Web JR 


务 目录 (如 ch03) 中 ,最 后 在 浏览 器 地 址 栏 输入 相应 的 地 址 访问 该 JSP 页 面 ,完成 效果 如 
图 3-1 所 示 。 


x) 


Qoi oe 


E 


图 3-1 JSP 文件 构成 页 面 效 果 


2) 使 用 Eclipse 完成 
按照 2. 2 节 介 绍 的 方法 ,在 Eclipse 中 建立 Web m AA JSP 文件 ,然后 编写 并 保存 代 
码 , 最 后 在 Eclipse 中 完成 运行 ,效果 同 图 3-1. 


3.1.3 拓展 训练 
1. 在 JSP 文 件 中 ,方法 的 定义 使 用 ( ) 符 号 。 


A. <% Wo B. <%! %> C. <%@ %> D. <%@ %> 
2. T£ JSP X fh Java 程序 片 使 用 ( MES. 
A. <% Wo B. «1 5» C. «AQ X> D. «G9 %> 


3. 在 Tomcat 服务 器 下 新 建 服务 目录 ch03, 打 开 记 事 本 ,输入 如 下 的 Test3_1. jsp 程 
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序 , 并 启动 Tomcat 服务 器 ,在 地 址 栏 中 输入 http: //localhost:8080/ch03/Test3_1.jsp 访问 
JSP 页 面 。 
代码 Test3_1. jsp iH F; 


<%@ page contentType= "text/html;charset— gb2312" ⁄> 
<html> 

<body bgcolor= "cyan" > 

<h3>R Bj 3k Z #l</h3— 

<!-- F BER PA 8k A BJ Jy A 09 ps Bl SERE 

<%! 

double sum(double a, double b) | 
return a+b; 
) 
%> 

二 %/* 下 面 是 方法 的 调用 过 程 ,并 且 注 释 的 内 容 在 客户 端 看 不 到 * / 062 

<%= sum(3.3,4.5)%> 

</body> 
</html> 


3.2 变量 和 方法 的 定义 


3.2.1 知识 要 点 


1. 变量 的 定义 

JSP 文件 中 ,可 以 在 “二 %!1” 和 “% 二 ”符号 之 间 定义 成 员 变量 ,变量 类 型 可 以 是 Java 请 
言 所 支持 的 任何 数据 类 型 。 例 如 下 面 的 例子 中 定义 了 两 个 普通 型 变量 i 和 j 初 值 都 为 0, 定 
义 了 一 个 双 精 度 实 型 变量 k 初 值 为 0.0, 定 义 了 一 个 字符 串 对 象 s 其 实体 为 “hello”。 


<%! 
int i=0,j=0; 
double k=0.0; 
String s= "hello"; 
n 


变量 的 定义 在 书写 位 置 上 没有 固定 要 求 ,但 习惯 上 总 是 将 其 放 在 JSP 文件 的 前 端 。 成 
员 变 量 一 旦 定义 就 在 整个 JSP 页 面 内 都 有 效 , 且 为 多 个 用 户 所 共享 。 也 就 是 说 ,任何 用 户 
操作 该 变量 都 会 对 其 他 用 户 产 生 影响 ,原理 如 图 3-2 所 示 。 例 如 3. 1. 2 节 中 的 JSP 文件 
eg3_1. jsp ,第 一 个 用 户 访问 页 面 时 显示 效果 为 "第 1 次 计算 ,结果 为 33. 0”, 第 二 个 用 户 再 访 
问 页 面 时 显示 效果 则 为 “第 2 次 计算 ,结果 为 33.0”, 以 此 类 推 ; 原因 就 在 于 成 员 变量 i 为 多 
个 用 户 所 共享 。 
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E 客户 线程 操作 
客户 [a TAFE 操作 
返回 结果 
请 求 
BP LI 客户 线程 _y 操作 
返回 结果 


H 


成 员 变 量 


图 3-2 共享 成 员 变量 

2. 方法 的 定义 

JSP 文件 中 ,也 可 以 在 “二 %1" 和 *“%" 符 号 之 间 定 义 方法 ,这 些 方法 将 在 Java ETUR 
中 被 调用 以 完成 相应 的 功能 ,关于 Java 程序 片 的 具体 内 容 会 在 下 一 节 中 详细 介绍 。 需 要 强 
调 的 是 ,这 些 方法 在 整个 JSP 页 面 内 都 有 效 , 但 方法 内 部 定义 的 变量 为 局 部 变量 只 在 该 方 
法 内 部 有 效 。 例 如 下 面 的 例子 就 是 定义 了 一 个 名 称 为 mul 的 方法 ,用 于 求 两 个 实数 的 乘 
积 ,返回 值 类 型 为 双 精 度 实 型 ,其 中 的 变量 a 和 b 为 局 部 变量 ,只 在 该 方法 内 部 有 效 , 离 开 了 
这 个 方法 , 便 不 能 再 被 访问 。 

<%! 

double mul(double a, double b) ( 


return a * b; 


> 


3.2.2 技能 操作 


编写 一 个 JSP 页 面 文件 eg3_2. jsp, 要 求 在 该 页 面 文件 中 定义 四 个 成 员 变 量 : resultl, 
result2、result3 和 result4( 初 值 均 为 0); 然后 定义 四 个 方法 ,分别 为 add( 求 两 个 数 的 和 )、 
sub( 求 两 个 数 的 差 )、mul( 求 两 个 数 的 积 ) 和 div( 求 两 个 数 的 商 )。 另 外 ,还 要 求 在 页 面 文件 
中 编写 一 段 Java 程序 片 ,在 Java 程序 片 中 调用 上 述 四 个 方法 以 实现 两 个 数 的 加 \ 减 、 乘 \ 除 
运算 ,并 将 最 终结 果 在 页 面 上 显示 出 来 。 

代码 模板 eg3_2. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset= GBK" 
pageEncoding="GBK" YG > 
<html> 
<head> 
<title> 2E Bt #ll Jy IE BJ E X — title > 
</head> 
<%! 
double result], result2, result3, result4 ; 
double add(double a, double b) ( 
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double c=a+b; 
return c; 
) 
double sub(double a, double b) ( 
double c—a— b; 
return c; 
) 
double mul(double a, double b) í 
double c—a * b; 
return c; 
) 
double div(double a, double b) í 
double c; 
if(b!—0) ( 
c=a/b; 
return c; 
) 
else return 0; 
) 
> 
<body bgcolor= "cyan" > 
<% 
resultl=add(3.5,1.2); 
result2=sub(3.5,1.2); 
result3=mul(3.5,1.2); 
result4=div(3.5,1.2); 
out. print(" 两 个 数 的 和 是 : "十 resultl 十 ","); 
out. print(" 两 个 数 的 差 是 : "十 result2 十 ","); 
out. print(" 两 个 数 的 积 是 :" 十 result3 十 ","); 
out. print(" 两 个 数 的 商 是 : "十 result4 十 "."); 
%> 
</body> 
</html> 


运行 效果 如 图 3-3 Bros 。 


http://localhost:8080/Test/eg3_2jsp 


3-3 变量 和 方法 的 定义 页 面 效 果 
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3.2.3 拓展 训练 
1. 编写 JSP 页 面 ,完成 显示 1 * 2 * 3 * ++ 10 乘积 的 功能 ,最 终 效 果 如 图 3-4 所 示 。 


http;//localhost:8080/Test/ 


E34 累 乘 页 面 效 果 
要 求 将 如 下 代码 模板 补充 完整 ,并 进行 调试 运行 。 


<%@ page contentType= "text/html;charset=GB2312"%> 
<html> 
<head> 
«title 5&2] 1</title> 
</head> 
<body bgcolor— "gray" 
【代码 段 1]// 定 义 长 整 型 变量 m, 初 值 为 1; 定义 整 型 变量 i, 初 值 为 1 
过 hl 二 计算 1 * 2 * 3*…*10 的 乘积 二 /hl 二 
<% 
while(i<=10)í 
【代码 段 2]// 给 出 累 乘 的 算法 
) 
%> 
结果 : <%=m> 
</body> 


</html> PPA] TER Ee RE 

2. 编写 JSP 页 面 , 完 成 求 和 及 求 平均 值 的 功能 ,并 GA 15.0 求 下 均 人 结果: no 
在 页 面 中 显示 ,最 终 效果 如 图 3-5 BER o 

要 求 将 如 下 代码 模板 补充 完整 ,并 进行 调试 运行 。 ” 图 35 求 和 及 求 平均 值 页 面 效果 


<%@ page contentType=" text/html; charset=GB2312"%> 
<html> 
<head> 
«title 4&2] 2</title> 

</head> 

<body bgcolor= "gray" > 

<%! 

float sum(float[ array) | 

float sum=0.0f; 
for(int i=0;i<array.length;i+-+) 
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( 
sum- — array[i] ; 
) 
return sum; 
) 
【代码 段 1]// 定 义 求 平均 值 的 方法 
»— 
<% 
float[] a= (1,2,3,4,5) ; 
float s sum(a) ; 
float result—aver(s, a. length) ; 
X> 
求 和 结果 : < =s%> 
求 平均 值 结果 : <% = result% > 
</body> 
</html> 


3.3 Java 程序 片 


3.3.1 知识 要 点 


JSP X fErp «np VL TES — 26 Wm Ya " EZ Rudi A. Java 程序 代码 ,使 得 JSP 页 面 的 功 
能 更 加 完备 ,这 些 程序 代码 在 JSP 页 面 中 被 称 为 Java 程序 片 或 Java 程序 段 。 一 个 JSP 页 
面 可 以 有 很 多 个 Java 程序 片 ,这 些 程序 片 将 被 JSP 引擎 ( 即 Tomcat 服务 器 ) 按 顺序 执行 。 
在 程序 片 中 声明 的 变量 为 局 部 变量 ,具体 的 作用 范围 与 其 声明 位 置 有 关 , 即 程序 片 中 的 局 部 
变量 从 声明 之 处 起 ,在 JSP 页 面 后 续 的 所 有 Java 程序 片 及 Java 表达 式 内 部 都 有 效 , 原 理 如 


图 3-6 所 示 。 
I remm 局 部 变量 
请 求 操作 Java 程 序 片 
客户 上 上 一 Caran D 局 部 变量 
返回 结果 


3-6 Java 程序 片 局 部 变量 


3.3.2 技能 操作 


编写 程序 产生 1 一 7 之 中 的 任意 一 个 随机 数 ,判断 随机 数 是 几 , 并 相应 地 输出 星期 几 。 
具体 步骤 如 下 : 
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。 用 JSP 声明 方式 产生 一 个 1 一 7 的 随机 数 方法 。 

。 用 JSP 程序 片 判断 产生 的 随机 数 是 几 。 

。 判断 以 后 在 页 面 中 相应 地 显示 出 星期 几 。 

打开 记事 本 ,输入 下 面 程序 ,并 保存 到 相应 的 Web 服务 目录 。 
代码 模板 eg3_3. jsp 如 下 : 


<%@ page language— "java" import= "java. util. * " pageEncoding= "GB2312" /4 > 
<html> 
<body bgcolor= "cyan"> 
<h1> 
二!-- 声 明 random 方面 ,产生 一 个 1 到 7 的 随机 数 -— 
<%! 
int random() | 
int i; 
Random r = new Random(); 
i=r. nextInt(7)+1; 
return i; 
) 
%> 
一 !-- 下 面 程序 是 对 random 方法 的 调用 ,并 判断 输出 星期 几 ==> 
<% 
int i = random(); 
out. print(" 本 次 访问 产生 的 随机 数 是 :" 十 i 十 "二 br 二 "); 
switch (i) í 
case 1: 
out. print(" 星 期 一 "); 


break; 

case 2: 
out. print(" BE HJ —") ; 
break; 

case 3: 
out. print(" 星 期 三 "); 
break; 

case 4: 
out. print(" 星 期 四 "); 
break; 

Case 5: 
out. print(" 星 期 五 "); 
break; 

case 6: 
out. print(" 星 期 六 "); 
break; 

case 7: 


out. print(" 星 期 日 "); 
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break; 
) 
»— 
</h1> 
</body> 
</html> 


运行 效果 如 图 3-7 所 示 。 


http://localhost:8080/ch03/eg3_3.jsp 


3-7 ”显示 星期 页 面 效果 


3.3.3 拓展 训练 


L 一 个 求 梯形 面积 的 方法 ,给 定 上 底 、 下 底 和 高 求 梯形 面积 ,最 终 运 行 效果 如 
图 3-8 E 


http://localhost:8080/Test/ 


3-8 求 梯形 面积 页 面 效果 
要 求 将 如 下 代码 模板 补充 完整 ,并 进行 调试 运行 。 


<%@ page contentType= "text/html;charset— gb2312" % > 
<html> 
<body bgcolor= "gray" > 
<h1> 
«HW! 
double area; 
double getArea(double a, double b, double c) í 
【代码 段 11// R BOE BJ 8 LAS A 
return area; 
) 
X> 
给 定 的 梯形 的 上 底 `、 下 底 和 高 分 别 是 : 3,5,4 
<br> 
面积 是 : 
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去 多 二 【代码 段 2]// 获 得 梯形 面积 
“> 
</h1> 

</body> 

</html> 


2. 模仿 上 面 的 例题 完成 求 三 角形 面积 的 功能 (注意 : 前 提 是 给 定 的 三 条 边 能 够 构成 三 
角形 ) 。 
3. 模仿 上 面 的 例题 完成 求 圆 形 面积 的 功能 。 


3.4 Java 表达 式 


3.4.1 知识 要 点 


JSP 文件 中 ,可 以 在 “二 %= 二 "和 “% 二 ”符号 之 间 插 入 Java 表达 式 , 这 个 表达 式 必 须 拥 
有 确定 的 值 ,其 值 的 计算 由 Web 服务 器 完成 ,计算 结果 会 以 字符 串 的 形式 发 送 到 客户 端 浏 
览 器 进行 显示 。 需 要 注意 的 是 ,“ 二 % 二 "和 “% 二 ”符号 之 间 不 能 插入 语句 只 能 插入 表达 式 ， 
而 且 “ 二 %==" 是 一 个 完整 的 符号 ,其 间 不 能 有 空格 ,“% 二 "符号 也 是 如 此 。 


3.4.2 技能 操作 


利用 Java 表达 式 在 JSP 页 面 中 输出 相应 的 内 容 。 具 体 任务 如 下 : 

* 借助 Math 类 中 提供 的 方法 ,输出 正弦 值 、. 输 出 立方 值 . 输 出 平方 根 值 ; 
。 输 出 算术 表达 式 的 值 .输出 逻辑 表达 式 的 值 。 

代码 模板 eg3_4. jsp MIT: 


<%@ page language= "java" contenttype— "text/html; charset=gbk" 
pageencoding= "gbk" %> 
<html> 
<head> 
<title>java Z5 XX </title> 
</head> 
<body bgcolor= "cyan" > 
<p>sin(3.14/6)% F 
<%= Math. sin(3.14/6)% > 
<p>8 的 立方 是 : 
<%=Math.pow(8,3) 967» 
<p>196 的 平方 根 等 于 
<% — Math. sqrt(196) X > 
<p>123456 乘 以 2 等 于 
<% —123456 * 2427 
«p210000 大 于 9999 吗 ?答案 : 
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<%=10000>9999%> 
</body> 
</html> 


运行 效果 如 图 3-9 所 示 。 
[arasan = muu 


http://localhost:8080/ch03/eg3_4jsp 


图 3-9 Java 表达 式 页 面 效果 


3.4.3 拓展 训练 


1. 利用 Java 程序 片 和 Java KIER. E JSP 页 面 中 输出 一 个 10 47 10 列 的 表格 ,每 一 个 
单元 格 中 显示 行列 坐标 值 ,最 终 运 行 效 果 如 图 3-10 所 示 。 


httpy//localhost8080/Test/ phan > Úm. | 


nmi piel WIBI mil unes titel WI nime uel pip] 
p gel [IIO] gu 115) RI 116] Iul PI NIS) 12110] wno 
Bi BR BIBI piu BIG Biel BIO 61118) B19) 61100) 
m WIRI WIBI miu | 115] tel MIN 118) 109) Hp] 
51107 1 15116) Bu 1511 [5] 511061 Siy] 51181 51101 [511 (101 
1610) OIG] 1611 D] 161114) 1611 [5] 1811161 加 上品 161118] 161109) 100) 
p miel "m miu "i5 "ie mi rm D71119) 100) 
181101) mpl [s] | B] md [811 [5] {811 16] IU! [8] [8] [s] | 9] {811 [10] 
JI Me IBI mul 191115] (ull) BI 191118] tI [91110] 
[0] [10] | 2) [10] B) 110) | (4) [10]1 [5] 110) | [6] noii [10] | (8) 0] [9] [10] | 00) 


3-10 ”表格 页 面 效 果 


要 求 编写 .调试 .运行 下 面 给 出 的 模板 代码 ,体会 html 语言 和 Java 程序 片 以 及 Java 表 
达 式 相互 之 间 的 协作 关系 。 


<%@ page language— "java" import= "java. util. * " pageEncoding— "UTF-8"%— 
<html> 
<head> 
<title> 10 fr 10 列 的 表格 一 /title> 
</head> 
<body> 
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<table cellpadding= "0" cellspacing= "0" border="1" width="100%"> 


<% 
int rows = 10; // 多 少 行 
int cols = 10; // 多 少 列 
for(int i = 0; i < rows; i++ ){ 
%> 
<tr align="center" height="30"> 
<% 
for(int j = 0; j < cols; j++ )( 
n 
«td [€ —ic1 9] | [<%=j+1 %>]</td> 
<% 
) 
) 
> 
</table> 
</body> 
</html> 


2. 综合 运用 变量 的 定义 Java 程序 片 和 Java 表达 式 等 所 学 内 容 , 完 成 在 JSP 页 面 中 输 


出 九 九 乘法 表 的 功能 ,最 终 效 果 如 图 3-11 所 示 。 


http://localhost:8080/Test/ >ý% 


图 3-11 九 九 乘法 表 页 面 效果 
要 求 将 如 下 代码 模板 补充 完整 ,并 进行 调试 运行 。 


<%@ page contentType= "text/html;charset = gb2312" 967» 
<html> 
【代码 段 1]// 定 义 整 型 变量 1 和 j 
<body bgcolor— "gray"2> 
<% 
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for(i=1;i<=9;i 十 十){ 
forG71;j€ — i; dt 
%> 
【代码 段 23/ /输出 乘法 表 内 容 
<%}%> 
<p> 
<%}%> 
</body> 
</html> 


3.5 JSP 指令 标记 


为 了 更 好 地 完成 Web 页 面 功能 ,JSP 引入 了 指令 标记 。 本 节 将 介绍 常用 的 两 个 指令 标 
id: page 指令 标记 和 include 指令 标记 。 


3.5.1 知识 要 点 


1. page 指令 标记 

page 指令 标记 用 来 定义 JSP 页 面 中 的 一 些 属性 及 其 属性 值 。 具 体 包 括 import 属性 、 
contentType 属性 、pageEncoding 属性 、language 属性 、session 属性 、isELIgnored 属性 (只 
限 JSP 2. 0), buffer 属性 、autoFlush JÑ PE, info 属性 、errorPage 属性 、isErrorPage 属性 、 
isThreadSafe 属性 .extends 属性 。 在 使 用 时 ,采用 如 下 格式 : 


一 %@page 属 性 1 一 "属性 1 的 值 "属性 2 一 "属性 2 的 值 ".… 属 性 na 一 "属性 n 的 值 " %> 
或 者 

二 %@page 属性 1=" 属 性 1 的 值 " 96 

二 %@page 属性 2 一 "属性 2 的 值 " %> 

二 %@page 属性 a 一 "属性 a 的 值 " 2 


在 本 书 中 只 介绍 import、contentType、pageEncoding 三 个 常用 属性 。 

1) import 属性 

使 用 page 指令 标记 的 import 属性 可 以 为 JSP 页 面 指定 应 该 引入 的 包 及 包 中 的 类 ,以 
便 JSP 页 面 在 变量 和 方法 的 定义 、Java 程序 片 Java 表达 式 中 正常 使 用 相应 包 中 的 类 。 

在 同一 个 JSP 文件 中 ,page 指令 标记 的 import 属性 可 以 多 次 出 现 ,也 就 是 说 ,可 以 指 
定 多 个 import 属性 值 ,导入 多 个 包 中 的 多 个 类 。 例 如, 下面 的 指令 : 


<%@ page import= "java. util. * , china. dalian. * " ⁄> 


表示 java. util 包 和 china. dalian 包 中 的 所 有 类 在 使 用 时 无 须 再 给 出 明确 的 包 标 识 符 。 
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另外 ,import 属性 的 书写 位 置 没有 特定 要 求 ,但 通常 情况 下 ,习惯 将 import 语句 放 在 
JSP 文件 顶部 附近 或 是 放 在 相应 的 包 首 次 使 用 之 前 。 

2) contentType 属性 

使 用 page 指令 标记 的 content Type 属性 可 以 用 来 设置 content Type 响应 报头 ,指明 即 
将 发 送 到 客户 端的 文档 的 MIME 类 型 (MIME 类 型 是 设 定 某 种 文件 用 匹配 的 一 种 应 用 程序 
来 打开 的 方式 类 型 ) 和 JSP 页 面 字符 的 编码 。 例 如 ,如 果 和 希望 客户 端 浏览 器 启用 HTML ft 
析 器 来 解析 执行 所 接收 到 的 信息 ,就 可 以 按照 如 下 方式 设置 content Type 属性 值 : 


<%@ page contentType= "text/html;charset=GB2312" % > 


如 果 和 希望 客户 端 浏览 器 启用 本 地 的 MS-Excel 应 用 程序 来 解析 执行 所 接收 的 信息 , 则 
可 以 按照 如 下 方式 设置 content Type 属性 值 : 


<%@ page contentType= "application/vnd. ms-excel" 267 


如 果 不 使 用 page 指令 标记 为 contentType 指定 属性 值 ,那么 content Type 属性 就 取 默 
认 值 , 即 "text/html;charset 王 ISO-8859-1"。 

与 import 属性 不 同 , 在 JSP 文件 中 ,page 指令 标记 只 能 为 contentType 属性 指定 一 个 
值 。 下 面 的 用 法 是 错误 的 : 

<%@ page contentType= "application/ vnd. ms-excel" %> 

— (à page contentType— "text/html;charset— GB2312" 267» 

可 以 利用 page 指令 标记 为 content Type 属性 指定 的 MIME 类 型 有 text/html GEB X K 
标记 语言 文本 ) text/plain (普通 文本 )、application/rtf (rtf 文本 ) , image/gif (gif 图 形 )、 
image/jpeg(jpeg 图 形 )、audio/basic(au 声音 文件 ) ,audio/midi C midi 音乐 文件 ) ,audio/x- 
pn-realaudio(realaudio 音乐 文件 ) , video/mpeg (mpeg 文件 )、video/x-msvideoCavi 文件 )、 
application/x-gzip(gzip 文件 ) .application/x-tar(tar 文件 ) ,application/vnd. ms-excel(excel 
文件 ) .application/msword(word 文件 ) 。 

3) pageEncoding 属性 

如 前 所 述 , 如 果 和 希望 同时 设置 所 接收 到 信息 的 MIME 类 型 和 JSP 页 面 字 符 编码 ,可 以 
使 用 contentType 属性 来 完成 。 但 是 ,如 果 只 想 更 改 字符 集 ,使 用 pageEncoding 属性 更 为 
简单 。 例 如 ,中 文 JSP 页 面 可 以 使 用 下 面 的 语句 : 


— (9 page pageEncoding— "GBK" %> 


来 设置 。 
关于 前 面 提 到 的 字符 集 , 经 常 使 用 的 有 如 下 几 种 : 
(D ASCII, 
ASCIICAmerican Standard Code for Information Interchange, 美 国信 息 互 换 标准 代 
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码 ) ,是 基于 常用 英文 字符 的 一 套 编码 。 

(2) ISO-8859-1, 

ISO-8859-1 编码 通常 叫做 Latin-1, 除 收录 ASCI 字符 外 ,还 增加 了 其 他 一 些 语言 和 地 
区 需要 的 字符 。 该 编码 是 Tomcat 服务 器 默认 采用 的 字符 编码 。 

(3) GB 2312, 

GB 2312 码 是 中 华人 民 共 和 国 国 家 标准 汉字 信息 交换 用 编码 ,简称 国标 码 , 是 由 国家 标 
准 总 局 发 布 的 关于 汉字 的 编码 ,通行 于 中 国 大 陆 和 新 加 坡 。 

(4) GBK. 

GBK 编码 规范 ,除了 完全 兼容 GB 2312, 还 对 繁体 中 文 和 一 些 不 常用 的 字符 进行 了 编 
码 。GBK 是 现 阶段 Windows 和 其 他 一 些 中 文 操作 系统 的 默认 字符 集 。 

(5) Unicode, 

Unicode 为 统一 的 字符 编码 标准 集 ,为 地 球 上 几乎 所 有 地 区 每 种 语言 中 的 每 个 字符 设 
定 了 统一 并 且 唯 一 的 编码 ,以 满足 跨 语言 . 跨 平台 进行 文本 转换 、 处 理 的 要 求 。 

(6) UTF-8, 

UTF-8 是 Unicode 的 一 种 变 长 字符 编码 。 用 在 网 页 上 可 以 在 同一 页 面 显 示 中 文 和 其 
他 语言 。 当 处 理 包 含 多 国文 字 的 信息 页 面 时 一 般 选择 用 UTF-8。 

2. include 指令 标记 

网 站 中 的 多 个 JSP 页 面 有 时 需要 显示 同样 的 信息 ,如 该 网 站 的 Logo 或 是 导航 等 ,为 了 
便于 网 站 的 管理 与 维护 ,通常 在 这 些 JSP 页 面 的 适当 位 置 嵌 入 一 个 相同 的 文件 以 完成 相应 
的 功能 。include 指令 标记 的 用 途 就 在 于 此 ,可 以 借助 该 标记 在 当前 JSP 页 面 中 整体 谋 入 另 
一 个 文件 。 语 法 格式 为 : 


<%@ include file— "3c f/F RJ kE" 967 
其 中 被 嵌入 的 文件 必须 是 可 访问 和 可 使 用 的 。 
3.5.2 技能 操作 


使 用 JSP 指令 标记 ,完成 相应 的 功能 。 具 体 任务 如 下 : 

* 利用 page 指令 标记 的 import 属性 ,导入 java. util 包 中 的 类 ,完成 在 当前 页 面 显示 
系统 时 间 的 功能 。 

。 利用 page 指令 标记 的 content Type 属性 ,指定 客户 端 浏 览 器 启用 本 地 的 Microsoft 
Word 应 用 程序 来 解析 执行 所 接收 的 信息 。 

* 利用 include 指令 标记 ,在 同一 网 站 的 不 同 页 面 顶 端 显示 相同 的 Logo 图 片 。 

1. 利用 page 指令 标记 的 import 属性 ,导入 java. util 包 中 的 类 ,完成 在 当前 页 面 显示 

系统 时 间 的 功能 。 
代码 模板 eg3_5. jsp 如 下 : 
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<%@ page language= "java" contentType= "text/html; charset— GBK" 4 > 
<%@ page import— "java. util. * "% > 
<html> 
<head> 
<title>JSP 指令 标记 <</title> 
</head> 
<%! 
String s=null ; 
> 
<body bgcolor= "cyan" > 
当前 的 日 期 和 时 间 是 : 
<% 
Date date=new Date() ; 
s = date. toString() ; 


An 

<%= s%> 
</body> 
</html> 


运行 效果 如 图 3-12 所 示 。 


二 二 BN > htpV/localhost8080/Tesyeg3_5jsp 


3-12. 显示 日 期 时 间 页 面 效 果 


2. 利用 page 指令 标记 的 contentType 属性 ,指定 客户 端 浏 览 器 启用 本 地 的 Microsoft 
Word 应 用 程序 来 解析 执行 所 接收 的 信息 。 
代码 模板 eg3_6. jsp MIT: 


— (8 page language— "java" contentType 一 "application/msword; charset= GBK" % > 
<html> 
<head> 
<title>JSP 指令 标记 < / title 

</head> 
<body bgcolor= "cyan" > 

到 P> 启 用 Microsoft Word 应 用 程序 处 理 所 接 收 到 的 信息 . 

<input type="text" size="12"> 
</body> 
</html> 


运行 上 述 页 面 时 会 弹出 如 图 3-13 所 示 的 “文件 下 载 ?对 话 框 , 单 击 “ 保 存 ” 按 钮 弹出 如 
图 3-14 所 示 的 对 话 框 ,选择 相应 的 路 径 保存 后 ,就 会 启用 本 地 的 Microsoft Word 应 用 程序 
来 显示 当前 页 面 内 容 , 如 图 3-15 所 示 。 
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名 称 : eg3_6.doc 
类 型 


: Microsoft Office Word 97 - 2003 文档 174... 
3f: localhost 


Cs 
Q Higgs 


图 3-13 “文件 下 载 ? 对 话 框 


[seamos ol 


图 3-14 选择 文件 保存 路 径 


3-15 Microsoft Word 解析 页 面 内 容 
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3. 利用 include 指令 标记 ,在 同一 网 站 的 不 同 页 面 顶端 显示 相同 的 logo 图 片 。 
代码 模板 eg3_7. jsp 如 下 : 


<%@ page language— "java" contentType= "text/html; charset=GBK"%> 
<html> 
<head> 
<title>Logo [8 H </title> 
</head> 
<body> 
<img src=" logo.jpg" > 
</body> 
</html> 


代码 模板 eg3_7_1.jsp 如 下 : 


<%@ page language— "java" contentType= "text/html; charset=GBK" %> 
<%@ include file—"eg3 7.jsp" %> 
<html> 
<head> 
<title>include 指令 标记 王 /title> 

</head> 
<body> 

<p> 

<font size="20" color="red" > 

立 警 为 公 执法 为 民 

</font> 
</body> 
</html> 


运行 上 述 代 码 , 显 示 效 果 如 图 3-16 所 示 。 其 中 上 半 部 分 的 图 片 来 自 于 被 包含 进来 的 
eg3_7.jsp 页 面 效果 ,下 半 部 分 的 文字 *“ 立 警 为 公 ”执法 为 民 "来 自 于 eg3_7_1.jsp 本 身 页 面 效果 。 


BL JP http//localhost808O/Test/eg3 7 1jsp 


立 警 为 公 执法 为 民 


图 3-16 include 指令 标记 应 用 之 一 
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代码 模板 eg3_7_2. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset=GBK" %> 
<%@ include file="logo.jsp" %> 
<html> 
<head> 
«itle include Jf $ br id < / title 
</head> 
<body> 
二 p 二 警察 (武警 或 人 民警 察 ) 
雪 p> 中 国 的 警察 包括 武警 和 人 民警 察 两 大 类 。 
二 p>“ 公 安 " 广 义 上 是 指 人 民警 察 , 分 为 公安 警察 .国家 安全 警察 及 司法 警察 。 
过 p 之 人 民警 察 是 国家 公务 员 ,实行 警 监 . 警 督 , 警 司 、 警 员 的 警衔 制度 ,服装 以 藏 黑 为 主 色调 。 
雪 p> 武 警 全 称 中 国人 民 武 装 警察 部 队 , 是 中 华人 民 共 和 国武 装 力量 的 一 部 分 ,是 担负 国家 赋予 的 
安全 保卫 任务 的 部 队 。 
</body> 
</html> 


运行 上 述 代码 ,显示 效果 如 图 3-17 Bros o 


图 22 httpJ//localhost:8080/Test/eg3_7_2jsp 


警察 武警 或 人 民警 察 ) 

中 国 的 警察 包括 武警 和 人 民警 察 两 大 类 。 

“公安 ”广义 上 是 指 人 民警 察 ， 分 为 公安 警察 、 国 家 安全 警察 及 司法 警察 。 

人 民警 察 是 国家 公务 员 ， 实 行 警 监 、 警 督 、 警 司 、 警 员 的 警衔 制度 ， 服 装 以 藏 黑 为 主 色 调 。 


武警 全 称 中 国人 民 武 装 警察 部 队 ， 是 中 华人 民 共 和 国武 装 力量 的 一 部 分 ， 是 担负 国家 赋 子 的 安全 保卫 任务 的 部 队 。 


图 3-17 include 指令 标记 应 用 之 二 


与 前 面 的 应 用 类 似 , 图 3-17 中 上 半 部 分 的 图 片 来 自 于 被 包含 进来 的 eg3_7.jsp 页 面 效 


果 , 下 半 部 分 的 文字 来 自 于 eg3_7_2. jsp 本 身 页 面 效果 。 
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3.5.3 拓展 训练 


1. 编写 一 个 JSP 页 面 ,利用 page 指令 标记 的 import 属性 将 java. util 包 中 的 类 、 
java. io 包 中 的 类 导入 ,然后 根据 实际 需求 进行 使 用 。 调 试 运行 程序 后 观察 页 面 效 果 。 

2. 把 3. 5.2 节 任 务 2 中 eg3 6.jsp 页 面 的 contentType 属性 值 修改 为 application/vnd 
. ms-powerpoint, 指 定 客户 端 浏览 器 启用 本 地 的 Microsoft PowerPoint 应 用 程序 来 解析 执 
行 所 接收 的 信息 。 运 行 修改 后 的 页 面 观察 页 面 效 果 。 

3. 使 用 “记事 本 ”编写 一 个 文本 文件 included. txt. included. txt 的 每 行 有 若干 个 英文 
单词 ,这 些 单词 之 间 用 空格 分 隔 ,每 行 之 间 用 一 br> 分 隔 ,具体 如 下 : 

included. txt 代码 : 

packag apple void back public 

<br> 


private throw class hello welcome 


编写 三 个 JSP 页 面 : first. jsp. second. jsp 和 third. jsp, 要 求 每 个 页 面 都 包含 included 
.txt 文件 ,其 余 页 面 内 容 自 行 设计 。 调 试 运 行程 序 后 观察 页 面 效 果 。 


3.6 JSP 动作 标记 


动作 标记 是 一 种 特殊 的 标记 ,它们 将 影响 JSP 运行 时 的 功能 。JSP 有 7 个 动作 标记 ,分 
别 是 : 

jsp:include 一 一 用 于 动态 引入 一 个 JSP 页 面 。 

jsp:param 一 一 用 于 传递 参数 ,必须 与 其 他 支持 参数 的 标签 一 起 使 用 。 

jsp:forward 一 一 执行 页 面 转向 ,将 请 求 的 处 理 转发 到 下 一 个 页 面 。 

jsp:plugin 一 一 用 于 下 载 JavaBean 或 Applet 到 客户 端 执行 。 

jsp:useBean 一 一 使 用 JavaBean。 

jsp:setProperty 一 一 修改 JavaBean 实例 的 属性 值 。 

jsp:getProperty 一 一 获取 JavaBean 实例 的 属性 值 。 

本 节 着 重 介绍 前 4 个 动作 标记 ,其 余 的 动作 标记 将 在 第 5 章 详细 介绍 。 
3.6.1 知识 要 点 

1. include 动作 标记 

include 动作 标记 用 于 在 当前 JSP 页 面 中 动态 包含 另 一 个 文件 ,即将 当前 JSP 页 面 、 被 


包含 的 文件 各 自 独立 地 翻译 为 字 节 码 文件 。 当 前 JSP 页 面 执行 到 该 动作 标记 时 , 才 加 载 执 
行 被 包含 文件 的 字 节 码 。 语 法 格式 如 下 : 
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<jsp:include page=" X f/F f Hb ab" / > 
或 是 


jsp:include 动作 标记 page 二 "文件 的 地 址 "二 

param 子 标记 

</jsp:include> 

需要 注意 的 是 , 当 include 动作 标记 不 需要 子 标记 时 ,必须 使 用 第 一 种 形式 。 在 3. 5 节 
中 介绍 的 指令 标记 与 本 节 的 include 动作 标记 功能 相似 ,都 是 将 另 一 个 文件 嵌入 到 当前 JSP 
页 面 中 ,但 是 其 处 理 include 指令 标记 方式 和 处 理 时 机 不 同 。include 指令 标记 是 静态 嵌入 ， 
即 在 编译 阶段 就 处 理 嵌 入 的 文件 ,被 嵌入 的 文件 在 语法 上 和 逻辑 上 依赖 于 当前 JSP 页 面 ， 
其 优点 是 执行 速度 快 ; 而 include 动作 标记 是 动态 嵌入 , 即 在 JSP 页 面 运行 时 才 处 理 嵌 入 的 
文件 ,被 蔡 入 的 文件 在 语法 上 和 逻辑 上 相对 独立 ,其 优点 是 可 以 借助 param 子 标记 灵活 处 
理 所 包含 的 文件 ,与 此 同时 也 会 造成 执行 速度 减 慢 。 

2. param 动作 标记 

param 动作 标记 用 于 设置 参数 值 ,这 个 动作 标记 本 身 不 能 独立 使 用 ,独立 的 param 动作 
标记 没有 实际 意义 , 需 作 为 jsp:include jsp:forward jsp:plugin 标记 的 子 标 记 来 使 用 ,以 完 
成 相应 的 功能 。 例 如 当 该 标记 与 jsp:include 动作 标记 一 起 使 用 时 ,可 以 将 param 标记 中 的 
值 传递 到 include 动作 标记 要 加 载 的 文件 中 ,被 加 载 的 JSP 文件 可 以 使 用 Web 服务 器 提供 
的 request 内 置 对 象 获取 include 动作 标记 的 param 子 标记 中 属性 所 提供 的 值 。 

语法 格式 如 下 : 


<jsp: param name= "paramName" value= "param Value" > 


3. forward 动作 标记 

forward 动作 标记 用 于 将 页 面 响应 转发 给 另外 的 页 面 , 既 可 以 转发 给 静态 的 HTML 页 
面 ,也 可 以 转发 到 动态 的 JSP 页 面 。 换 言 之 ,就 是 从 该 动作 标记 处 停止 执行 当前 JSP 页 面 ， 
而 转向 指定 的 HTML 页 面 或 JSP 页 面 去 执行 。 语 法 格式 如 下 : 


<jsp:forward page 一 "将 转向 的 页 面 "/ 二 
或 是 


<jsp:forward page 一 "将 转向 的 页 面 "之 
param 子 标记 
</jsp:forward> 
需要 注意 的 是 , 当 forward 动作 标记 不 需要 子 标记 时 ,必须 使 用 第 一 种 形式 。 
4. plugin 动作 标记 
plugin 动作 标记 用 于 执行 一 个 applet (Java 小 应 用 程序 ), 因 为 有 些 浏览 器 并 不 支持 
Java applet 程序 ,所 以 可 以 借助 plugin 动作 标记 保证 客户 端 浏览 器 能 够 顺利 执行 Java 
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applet 程序 。 在 具体 执行 过 程 中 ,plugin 动作 标记 指示 JSP 页 面 加 载 Java plugin, 该 插件 由 
客户 负责 下 载 , 并 使 用 该 插件 来 运行 Java applet 程序 。 语 法 格式 如 下 


jsp: plugin 
type= "applet " 
code=" F Im xc f" 


codebase— "F 45 83 3c f/F PR 48 " 

[ height 一 "小 程序 高 度 值 " ] 

[ width 一 "小 程序 宽度 值 " ] 

[ jreversion 一 "JRE 版 本 号 " ] 

[ isp:fallback2 zs JH P ff] KANA B — / jsp: fallback> ] 
—/jsp: plugin 


下 面 的 例子 代码 ,表明 小 应 用 程序 的 字 节 码 文件 是 A. class, 且 该 字 节 码 文件 存放 在 当 
前 Web 服务 目录 中 ; 小 应 用 程序 显示 高 度 为 180 像素 ,宽度 为 200 像素 ,JRE 版 本 号 为 
1.1; 当 插件 不 能 启动 时 显示 给 用 户 一 段 提 示 性 文字 “Unable to load applet", 

二 jsp:plugin 


type 一 "applet " 
code="A. class" 


codebase="." 
height— "180" 
width— "200" 


jreversion— "1.1" 
«jsp :fallback Unable to load applet. < /jsp:fallback- 
—/jsp: plugin > 
需要 注意 的 是 ,上 面 语法 格式 中 被 *[]” 括 起 来 的 部 分 为 可 选项 ,可 以 根据 实际 情况 来 决 
定 是 和 否 对 其 进行 设置 。 标 记 一 jsp:fallback 二 是 一 jsp:plugin 二 动作 标记 的 一 部 分 ,并 且 只 能 
在 二 jsp:plugin 过 动作 中 使 用 。 


3.6.2 技能 操作 


使 用 JSP 动作 标记 ,完成 相应 的 功能 。 具 体 任务 如 下 : 
。 利 用 include 动作 标记 和 param 动作 标记 完成 简单 的 欢迎 登录 页 面 。 
。 利用 forward 动作 标记 和 param 动作 标记 实现 奇偶 页 的 不 同 跳 转 页 面 。 
1. 利用 include 动作 标记 和 param 动作 标记 完成 简单 的 欢迎 登录 页 面 。 
代码 模板 eg3_8. jsp 如 下 : 
<%@ page language— "java" contentType= "text/html; charset=GB2312" ⁄> 
<html> 

<head> 


<title>JSP HER </title> 
</head> 
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<body bgcolor= "cyan" > 

<% 

String user= "sa", password= "123456"; 

x» 

«jsp:include page= "success.jsp"— 
—jsp: param name= "textl" value— "— Yi — user %>"/> 
jsp: param name— "text2" value— "< % = password % >"/> 

—/jsp:include- 

</body> 
</html> 


代码 模板 success. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset=GBK"%> 
登录 成 功 ,欢迎 

<br> 

您 的 用 户 名 是 : 

<% — request. getParameter("textl") Yi > 

<br> 

您 的 密码 是 : 

<% — request. getParameter( "text2") % > 


最 终 运行 结果 如 图 3-18 所 示 。 可 见 ,eg3_8.jsp 中 通过 下 面 两 行 语句 : 


一 jsp:param name 一 "textl" value— " — Yi — user %>"/> 

<jsp:param name 一 "text2" value— "< % = password % >"/> 

设 定 了 传递 的 参数 列表 , 即 名 称 为 textl 的 参数 其 值 为 
user 变量 的 值 ,也 就 是 "sa" 字 符 串 ; 名 称 为 text2 的 参数 其 值 为 
password 变量 的 值 ,也 就 是 "123456" 字 符 串 。 

2. 利用 forward 动作 标记 和 param 动作 标记 实现 奇偶 页 


的 不 同 跳 转 页 面 。 图 3-18 欢迎 登录 页 面 效果 
代码 模板 eg3_9. jsp 如 下 : 


http://niudy-pc:8080/eg/ 


— (8 page language— "java" contentType= "text/html; charset— GBK" 967 
<html> 
<body> 
<% 
int number= (int) (Math. random() * 1007-1) ; 
if(number942! —0)( 
X> 
<jsp:forward page= "eg3_9_1.jsp"> 
—jsp: param name= "text" value= "< % =number% >"/> 
</jsp:forward> 
<% 
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) 
else 
( 
X> 
<jsp:forward page= "eg3_9_2.jsp"> 
—jsp: param name= "text" value= "< % = number% >"/> 
</jsp:forward> 
«Hu 
“> 
</body> 
</html> 


代码 模板 eg3_9_1. jsp 如 下 : 


<%@ page language— "java" contentType— "text/html; charset=GBK"%> 
<html> 

<body bgcolor= "cyan"> 

传递 过 来 的 随机 数 为 : 

<% — request. getParameter( "text") % > 

<p> 

该 奇数 页 图 片 为 : 

<p> 

<img src— "D:Npicture1.jpg"> 

</body> 


代码 模板 eg3_9_2. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset= GBK" % > 
<html> 

<body bgcolor= "cyan" > 

传递 过 来 的 随机 数 为 : 

<% — request. getParameter("text") ⁄ > 
<p> 

该 偶数 页 图 片 为 : 

p 

<img src— "D: Vpicture2.jpg"— 
</body> 

</html> 


具体 运行 效果 如 图 3-19 和 图 3-20 所 示 。 从 图 中 可 以 看 出 , 当 eg3 9. jsp 中 产生 的 随机 


数 为 奇数 时 , 则 跳 转 到 eg3_9_1. jsp 页 面 , 显 示 图 像 为 人 物 左 脸 ; 当 eg3_9. jsp 中 产生 的 随 
机 数 为 偶数 时 , 则 跳 转 到 eg3_9_2. jsp 页 面 ,显示 图 像 为 人 物 右 脸 。 


56 网 络 编程 与 动态 网 站 制作 


http://niudy-pc:8080/eg/ http://niudy-pc:8080/eg/ 


图 3-19 奇数 页 跳 转 页 面 效 果 图 3-20 ”偶数 页 跳 转 页 面 效果 


3.6.3 拓展 训练 


1. 编辑 下 面 给 出 的 代码 模板 ,调试 运行 并 观察 页 面 效 果 , 体 会 include 指令 标记 和 
include 动作 标记 的 联系 与 区 别 。 
代码 模板 static. html 如 下 


— (3 page language— "java" contentType= "text/html; charset=GBK"%> 
<html> 
<head> 
<title>JSP 动作 标记 <</title> 
</head> 
<body bgcolor= "cyan" > 
name: input type="text" value="123"> 
password: input type="text" value—"123"7 
<input type="button" value="login" name= "button" > 
</body> 
</html> 


代码 模板 dynamic. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset— GBK" /4 7 
<html> 
<head> 
<title>JSP ali fEbs id < title 
</head> 
<body> 
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我 来 自 第 2 个 文件 . 
</body> 
</html> 


代码 模板 index. jsp 如 下 : 


<%@ page language— "java" contentType= "text/html; charset=GBK"%> 
<html> 
<head> 
<title>JSP 动作 标记 所 /title> 
</head> 
<body bgcolor= "cyan"> 
这 是 include 指令 标记 举例 . 
<br><%@ include file— "static. html" 267 


br X Jš include 动作 标记 举例 . 
«br —jsp:include page= "dynamic.jsp" flush "true" /7- 
</body> 
</html> 


2. 模仿 3.6.2 节 中 的 例题 ,使 用 forward 动作 标记 和 param 动作 标记 实现 页 面 跳 转 功 
能 。 要 求 : 编写 一 个 静态 HTML 文件 decide. html ,在 该 文件 中 定义 一 个 变量 i 并 为 其 赋 初 
值 , 当 i 的 值 大 于 0 时 , 跳 转 到 页 面 positive. jsp; 否则 跳 转 到 页 面 negative. jsp. positive 
.jsp 和 positive. jsp 页 面 内 容 可 以 自行 设计 。 


3.7 1 2 


一 个 JSP 页 面 通常 由 普通 的 HTML 标记 、JSP 注释 、Java 动态 元 素 ( 包 括 变量 和 方 
法 的 声明 Java 程序 片 Java 表达 式 ) 以 及 JSP 标记 (包括 指令 标记 、 动 作 标记 和 自 
定义 标记 ) 组 成 。 

在 一 个 Java 程序 片 中 声明 的 变量 称 为 JSP 页 面 的 局 部 变量 ,它们 在 JSP 页 面 后 续 
的 所 有 程序 片 部 分 以 及 表达 式 部 分 有 效 。 一 个 用 户 对 JSP 页 面 局 部 变量 操作 的 结 
A ,不 会 影响 到 其 他 用 户 。 

JSP 页 面 成 员 变量 是 被 所 有 用 户 共 享 的 变量 ,任何 用 户 对 JSP 页 面 成 员 变 量 操作 的 
结果 ,都 会 影响 其 他 用 户 。 

page 指令 标记 用 来 定义 整个 JSP 页 面 的 一 些 属性 以 及 这 些 属性 的 值 。page 指令 只 
能 为 content Type 属性 指定 一 个 值 . 但 可 以 为 import 属性 指定 多 个 值 。 

include 指令 标记 是 先 将 当前 JSP 页 面 与 要 财 入 的 文件 合并 成 一 个 新 的 JSP 页 面 ， 
然后 再 由 JSP 引擎 将 新 页 面 转化 为 Java 文件 处 理 并 运行 。 而 include 动作 标记 在 
JE JSP 页 面 转译 成 Java 文件 时 ,并 不 合并 两 个 页 面 ; 而 是 在 Java 文件 的 字 节 码 文 
件 被 加 载 和 执行 时 , 才 去 处 理 include 动作 标记 中 引入 的 文件 。 


JSP 内 置 对 象 


为 简化 Web 页 面 的 开发 ,JSP 本 身 提 供 了 一 些 内 置 对 象 。 这 些 内 置 对 象 在 JSP 页 面 中 
不 需要 声明 和 实例 化 ,就 可 以 直接 在 Java 程序 片 和 Java 表达 式 中 使 用 ,它们 由 Web 服务 器 
责 实现 和 管理 。 
JSP 提供 的 内 置 对 象 有 request 对 象 response 对 象 session X% application 对 象 out 
对 象 .pageContext 对 象 config 对 象 .exception 对 象 。 本 章 将 主要 介绍 前 4 个 常用 内 置 对 
象 的 使 用 方法 ,其 他 内 置 对 象 及 其 常用 方法 可 参考 本 书 附录 C. 


4.1 request 对 象 


4.1.1 知识 要 点 


request 对 象 代表 请 求 对 象 , 它 是 实现 了 javax. servlet. ServletRequest 接口 的 一 个 实 
例 。 当 用 户 请 求 一 个 JSP 页 面 时 ,JSP 页 面 所 在 的 服务 器 将 用 户 发 出 的 所 有 请 求 信息 封装 
TE VI BL X request 中 ,使 用 该 对 象 就 可 以 获取 用 户 提交 的 信息 。 

下 面 就 着 重 学 习 request 对 象 的 两 个 常用 方法 , 即 getParameter(String name) 方 法 和 
getParameterValues(String name) 方 法 ,它们 的 主要 功能 是 用 来 获取 客户 提交 信息 。 

1. public String getParameter(String name) 方 法 

该 方法 以 字符 串 的 形式 返回 客户 提交 的 信息 ,而 客户 向 Web 服务 器 提交 信息 的 方式 通 
常 是 使 用 HTML 表单 来 进行 的 ,其 一 般 格式 为 : 

<form method 二 "get"|"post"action 二 "提交 信息 将 要 传送 到 的 目标 地 址 "二 

提交 手段 (包括 文本 框 \ 文 本 区 、 下 拉 列 表 、 单 选 按钮 . 复 选 框 等 ) 

</form> 

Hp < form ERAPR , method 取 值 get 或 post, get 方法 和 post 方法 的 主要 区 别 
是 : 使 用 前 者 提交 的 信息 会 在 提交 过 程 中 显示 在 浏览 器 地 址 栏 中 ,而 使 用 后 者 提交 的 信息 
不 会 显示 在 地 址 栏 中 。 例 如 : 


<form method= "post"action= "destination. jsp"> 
<input type="text" name= "loginName" value="sa"/> 
<input type— "submit" name= "button" value— "1E JE" /— 
</form> 
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单 击 “注册 ”按钮 后 ,JSP 页 面 destination. jsp 就 可 以 借助 下 面 的 代码 : 
String namelnfo = request. getParameter("loginName"); 


来 获得 客户 端 提交 的 文本 框 中 预先 设 定 的 信息 “sa”, 并 进行 后 续 的 处 理 和 操作 , 且 提 交 的 信 
息 不 会 显示 在 浏览 器 地 址 栏 中 。 

2. public String[ ] getParameterValues(String name) 

该 方法 以 字符 串 数组 的 形式 返回 客户 端 向 服务 器 端 传递 的 指定 参数 名 的 所 有 值 。 
例如 : 


<form method= "post"action= "destination. jsp"> 
您 的 兴趣 爱好 : 
<input type= "checkbox" name= "hobbies" value= "reading" /> [i] i£ 
<input type= "checkbox" name= "hobbies" value= "music" /7 k 
<input type="checkbox" name= "hobbies" value= "surfing" /7» E W 
" value= "swimming" /> llf jk 
" value=" badminton" /7 H E X 


<input type="checkbox" name= "hobbies 
<input type= "checkbox" name= "hobbies 
</form> 


如 果 用 户 选 择 了 “阅读 ”“ 音 乐 "“ 上 网 ”和 “游泳 ”四 项 兴趣 爱好 ,那么 提交 表单 后 JSP 
页 面 destination. jsp 就 可 以 借助 request 内 置 对 象 的 getParameterValues 方法 获取 用 户 提 
交 的 信息 , 即 复 选 框 hobbies 的 属性 值 reading, music, surfing 和 swimming ,并 将 信息 存放 
到 一 个 数组 中 ,该 数组 的 元 素 的 值 与 复 选 框 的 属性 值 一 一 对 应 。 具 体 代码 如 下 : 


String hobbies[ ] = request. getParameterValues("hobbies"); 


4.1.2 技能 操作 


使 用 request 内 置 对 象 获取 客户 提交 的 信息 。 具 体 任务 如 下 : 

编写 两 个 JSP 页 面 eg4 1. jsp 和 destination. jsp.eg4 1. jsp 通过 表单 向 destination. jsp 
提交 输入 的 用 户 名 和 选择 的 兴趣 爱好 ,destination. jsp 负责 获得 表单 中 提交 的 信息 并 显示 。 

代码 模板 eg4_1. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"% > 
<html> 
<head> 
<title> eg4_1.jsp</title> 
</head> 
<body bgcolor= "cyan" > 
<form action= "destination. jsp" method= "post" > 
用 户 名 : <input type="text" name= "loginName"/> 
<br> 
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您 的 兴趣 爱好 : 
<input type="checkbox" name= "hobbies" value 二 "reading"/ 记 阅读 
<input type="checkbox" name= "hobbies" value= "music" /7 Bk 
<input type="checkbox" name= "hobbies" value= "surfing" />> E R 
<input type="checkbox" name= "hobbies" value= "swimming" / 7 llf vk 
<input type="checkbox" name= "hobbies" value=" badminton" / 3] E X 
<br> 
<input type="submit" value— "iE H "/> 
</form> 
</body> 
</html> 


代码 模板 destination. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 

"GBK"%> 

<html> 

<head> 

<title>getValue.jsp< /title> 

</head> 

<body bgcolor=" cyan"> 

<% 
String namelnfo= request. getParameter("loginName"); // 获 得 eg4 1.jsp 页 面 中 输入 的 姓名 
String hobbies 口 = request. getParameterValues("hobbies") ; // 获 得 eg4 1.jsp 页 面 中 选择 的 兴趣 


“> 
您 输入 的 姓名 是 : <% = nameInfo 267 — br 
您 的 兴趣 爱好 有 : 
<% 
for(int i=0;i<hobbies.length;i+ +) ( 
out. print(hobbies[i] +" ");) 
“> 
</body> 
</html> 


最 终 运 行 页 面 效 果 如 图 4-1 和 图 4-2 所 示 。 


http://localhost:8080/ch04/eg4_1jsp ` 


41 例子 4_1 页面 效果 之 一 
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http;//localhost:8080/ch04/destinationjsp?loginName-sai ~ 


图 4-2 例子 4_1 页 面 效 果 之 二 


4.1.3 拓展 训练 


1. 如 果 在 上 述 eg4_1. jsp 页 面 中 不 选择 任何 兴趣 爱好 ,而 直接 单 击 “ 注 册 ” 按 钮 ,观察 
destination. jsp 页 面 中 会 出 现 什么 结果 ? 在 你 的 浏览 器 中 是 否 出 现 如 图 4-3 所 示 的 信息 , 即 
NullPointerExceptionCZ: $ £F) 5t 26, 要求 : 按照 下 面 的 提示 解决 eg4 _1. jsp 中 的 
NullPointerException( 空 指针 ) 异 常 问题 。 


http://localhost:8080/ch04/destinationjsp?loginName- ~ 


HTTP Status 500 - 


[STE Exception report 
CETT The server encountered 


org.apache.jasper.JasperException: An exception occurred processing JSP page /destina| 


11: 您 输入 的 姓名 是 : «x-nameInfo *><br> 
12: 您 的 兴趣 爱好 有 : 


43 空 指针 异常 提示 


为 了 避免 在 运行 时 出 现 NullPointerException 异常 ,可 以 在 destination. jsp 页 面 中 显 
示 兴 趣 爱好 部 分 代码 修改 为 : 
if( hobbies! = null) ( 
for(int i=0;i<hobbies.length;i+- +) ( 
out. print(hobbies[i] +" "); 
) 
) 


如 此 修改 后 ,即使 用 户 没有 对 兴趣 爱好 进行 任何 选择 ,也 不 会 出 现 上 面 提 到 的 空 指 针 异 
常 了 ,最 终 的 显示 效果 如 图 4-4 所 示 。 这 样 操作 ,有 效 保证 了 JSP 页 面 良好 的 交互 性 。 

2. 如 果 在 eg4_1. jsp 页 面 的 文本 框 中 输入 中 文 姓名 ,观察 destination. jsp 页 面 中 获得 
的 用 户 名 是 什么 效果 ? 在 你 的 浏览 器 中 是 否 出 现 如 图 4-5 所 示 的 乱码 (用 户 名 由 “?” 等 组 
成 )。 要 求 : 按照 下 面 的 提示 解决 egi 1.jsp 中 的 中 文 乱码 问题 。 
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http://localhost:8080/ch04/destinationjsp?loginName= http://localhost:8080/ch04/destinationjsp 


图 4-4 修改 后 的 例子 4_1 页 面 效 果 图 4-5 中 文 乱码 


解决 乱码 常用 的 方法 有 两 种 : 

(1) 使 用 setCharacterEncoding(String code) 设 置 统一 字符 编码 。 

request 对 象 提供 了 方法 setCharacterEncoding(String code) 设 置 编码 ,其 中 参数 code 
以 字符 串 形 式 传 人 要 设置 的 编码 格式 ,但 这 种 方法 仅 对 于 提交 方式 是 post 的 表单 有 效 ( 表 
单 默认 的 提交 方式 是 get)。 例 如 ,我 们 使 用 该 方法 解决 eg4_1. jsp 和 destination. jsp 页 面 
中 出 现 的 中 文 乱码 问题 ,需要 完成 两 件 事 : 

首先 ,确保 eg4_1.jsp 中 的 表单 提交 方式 为 post, 具 体 代码 如 下 : 


<form action= "destination. jsp" method="post"> 
其 次 ,在 destination. jsp 中 获取 表单 信息 之 前 设置 统一 编码 ,具体 代码 如 下 : 
request. setCharacterEncoding( " GBK") ; 


(2) 对 获取 的 信息 进行 重新 编码 。 

通过 内 置 对 象 request 获取 到 字符 串 的 值 后 ,对 该 字符 串 使 用 ISO-8859-1 重新 编码 ,并 
把 编码 的 结果 存放 到 一 个 字 节 数组 中 ,然后 再 把 这 个 字 节 数组 转化 为 字符 串 。 例 如 ,使 用 该 
方法 解决 eg4_1. jsp 和 destination. jsp 页 面 中 出 现 的 中 文 乱 码 问 题 , 具 体 代 码 如 下 : 

String name= request. getParameter("loginName") ; 


byte b[ ] = name. getBytes("ISO-8859-1") ; 
name- new String(b) ; 


4.2 response 对 象 


4.2.1 知识 要 点 


当 用 户 请 求 服务 器 的 一 个 页 面 时 ,会 提交 一 个 HTTP 请 求 , 服 务 器 收 到 请 求 后 ,返回 
HTTP 响应 。4. 1 节 中 已 经 介绍 过 .request 对 象 能 够 将 请 求 信息 进行 封装 ,而 本 节 中 将 要 
学 习 的 response 对 象 能 够 为 用 户 的 请 求 做 出 动态 响应 。 可 以 说 ,request 对 象 与 response 
对 象 是 相互 匹配 的 两 个 对 象 。 下 面 就 是 response 对 象 做 出 动态 响应 的 三 个 方面 的 内 容 。 

1. 动态 响应 contentType 属性 

JSP 页 面 用 page 指令 标记 设置 了 页 面 的 contentType 属性 值 response 对 象 就 按照 这 
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种 属性 值 的 方式 对 客户 做 出 响应 。 第 3 章 已 经 介绍 过 ,在 page 指令 中 只 能 为 contentType 
属性 指定 一 个 值 , 所 以 如 果 想 动态 指定 contentType 属性 值 , 换 一 种 方式 来 响应 客户 ,可 以 
让 response 对 象 调 用 setContentType(String s) 方 法 来 重新 设置 网 页 响应 的 MIME 类 型 。 
常见 的 MIME 类 型 有 text/html ( 超 文 本 标记 语言 文本 )、text/plain (普通 文本 )、 
application/rtf(rtf XÆ) ,image/gifCgif 图 形 ) ,image/jpegCGpeg 图 形 ) ,audio/basic(au 声音 
文件 ) 、audio/midi(midi 音乐 文件 ) ,audio/ x-pn-realaudio(realaudio 音乐 文件 ) , video/mpeg 
(mpeg 文件 ) .video/x-msvideo(avi 文件 )、application/x-gzip(gzip 文件 )、application/x-tar 
(tar 文件 ) .application/vnd. ms-excelCexcel 文件 ) .application/msword(word 文件 ) 等 。 
例如 ,下 面 的 语句 : 


response. setContentType( "application/msword; charset— GB2312") ; 


将 会 提示 用 户 启用 Microsoft Word 来 显示 或 保存 当前 页 面 。 

2. 设置 响应 表 头 (HTTP 文件 头 ) 

response 对 象 可 以 通过 方法 set Header (String name, String value) 设 置 指定 名 字 的 
HTTP 文件 头 的 值 , 以 此 来 操作 HTTP 文件 头 。response 对 象 设置 的 新 值 将 会 覆盖 原 值 。 
如 果 和 希望 某 页 面 每 5 秒 钟 刷新 一 次 ,那么 应 在 该 页 面 中 添加 如 下 代码 : 


response. setHeader( "refresh", "5"); 


3. response 重 定向 

在 需要 将 用 户 引 导 至 另 一 个 页 面 时 ,可 以 使 用 reponse 对 象 的 sendRedirect (String 
url) 方 法 实现 用 户 的 重 定向 。 例 如 用 户 输入 的 表单 信息 不 完整 ,应 该 再 次 被 重 定向 到 输入 
页 面 。 例 如 下 面 的 语句 ,会 重 定向 到 input. jsp 页 面 : 


response. redirect("input.jsp") ; 


4.2.2. 技能 操作 


使 用 response 内 置 对 象 动态 响应 客户 的 请 求 ,具体 任务 如 下 : 

。 动态 响应 contentType 属性 值 。 

. 设置 响应 表 头 CHTTP 文件 头 ) 。 

° 重 定向 到 指定 页 面 。 

1. 动态 响应 contentType 属性 值 

编写 JSP 页 面 eg4_2. jsp. 客户 通过 单 击 页 面 上 的 不 同 按钮 ,可 以 改变 页 面 响应 的 
MIME 类 型 。 当 单 击 Microsoft Word 按钮 时 ,JSP 页 面 动态 地 改变 contentType 的 属性 值 
为 application/msword, 客 户 浏览 器 启用 本 地 的 Word 软件 来 显示 当前 页 面 内 容 ; 当 单 击 
Microsoft Excel 按钮 时 .JSP 页 面 动态 地 改变 content Type 的 属性 值 为 application/vnd. 
ms-excel, 客 户 浏 览 器 启用 本 地 的 Excel 软件 来 显示 当前 页 面 内 容 ; 当 单 击 Microsoft 
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PowerPoint 按钮 时 ,JSP 页 面 动 态 地 改变 content Type 的 属性 值 为 application/msword, 客 
户 浏 览 器 启用 本 地 的 PowerPoint 软件 来 显示 当前 页 面 内 容 。 
代码 模板 eg4_2. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK" A42 
<html> 
<head> 
<title> eg4_2.jsp</title> 
</head> 
<body bgcolor= "cyan"> 
<form action= ""method= "post" > 
<p>{EJH response 动态 响应 contentType 属性 值 : 
<p> B Microsoft Word 按钮 ,使 用 Word 显示 页 面 内 容 
<input type="submit" value= "Word" name= "submit"> 
<p> fil Microsoft Excel 按钮 ,使 用 Excel 显示 页 面 内 容 
< input type="submit" value= "Excel" name= "submit" 
<p> B Microsoft PowerPoint 按钮 ,使 用 PowerPoint 显示 页 面 内 容 
<input type="submit" value= "PowerPoint" name= "submit"> 
<% 
String str = request. getParameter( "submit" ) ; 
if ("Word". equals(str)) { 
/ / response 调用 setContentType 方法 设置 MIME 类 型 为 application/msword 
response. setContentType("application/msword") ; 
) elseif ("Excel".equals(str)) ( 
/ / response 调用 setContentType 方法 设置 MIME 类 型 为 application/ vnd. ms-excel 
response. setContentType( "application/ vnd. ms-excel") ; ) 
n 
</form> 
</body> 
</html> 


最 终 运 行 页 面 效 果 如 图 4-6 一 图 4-8 所 示 。 


http;//localhost:8080/ch04/eg4 2 jsp. 


图 4-6 例子 4_2 页 面 效果 
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AaBbCcD, ñaBbcop. AaBt 
(Ex o ME 


4-8. 使 用 Excel 显示 页 面 内 容 
2. 设置 响应 表 头 (HTTP 文件 头 ) 。 


编写 一 个 JSP 页 面 eg4_3. jsp, 在 该 页 面 中 使 用 response 对 象 设置 一 个 响应 头 refresh, 
其 值 是 5。 那 么 用 户 收 到 这 个 头 之 后 ,该 页 面 会 每 5 秒 钟 刷新 一 次 。 
代码 模板 eg4_3.jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBI 


" pageEncoding = 
"GBK"%> 
<%⁄@ page import= "java.util. * " ⁄> 
<html> 
<head> 


<title>eg4_3.jsp</title> 
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</head> 
<body> 
—<h2> i JSP 页 面 每 隔 5 秒 钟 就 会 刷新 1 K, Bi tE SS l Se pria] ë fe / h27 
<p> BJ BEP BREST E : 
<% 
Date d=new Date(); 
out. print("" +d. getSeconds()) ; 
response. setHeader(" refresh" , "5") ; 
// Fl response 对 象 设 置 一 个 响应 头 “refresh” ,其 值 是 “3”. 
%> 
</body> 
</html> 


最 终 运 行 页 面 效 果 如 图 4-9 所 示 。 


http;//localhost:8080/ch04/eg4 3.jsp 


图 4-9 ”例子 4_3 页 面 效 果 之 一 
当 5 秒 钟 过 后 ,该 JSP 页 面 会 自动 刷新 ,出 现 如 图 4-10 所 示 的 页 面 效 果 。 


Wiocalhost8080/ch04/eg4 .3jsp 


4-10 例子 4_3 页 面 效果 之 二 


3. 重 定向 到 指定 页 面 

编写 两 个 JSP 页面, 即 eg4_4. jsp 和 verify. jsp, 如 果 在 页 面 eg4_4. jsp 中 输入 正确 的 管 
理 员 账 号 sa 以 及 管理 员 密码 admin123, 单 击 “ 登 录 ” 按 钮 后 将 账号 和 密码 信息 提交 给 页 面 
verify. jsp; 如 果 输 入 不 正确 , 则 重新 定向 到 eg4_4. jsp 页 面 。 

代码 模板 eg4_4.jsp 如 下 : 


<% (B page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title>eg4_4.jsp</title> 
</head> 
<body bgcolor= "cyan"> 
<form action= "verify. jsp" method="post" name= "form"> 


所 示 。 


CBP sa) 和 正确 的 管理 员 密码 ( 即 admin123) ,并 
单 击 “登录 ”按钮 后 ,会 出 现 如 图 4-12 和 图 4-13 
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请 输入 管理 员 账 号 : 
< input type="text" name= "text"> 
<p> 
请 输入 管理 员 密 码 : 
<input type="password" name— "pwd"/> 
<input type="submit" value 一 "登录 "二 
</form> 
</body> 
</html> 


代码 模板 verify. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title>verify.jsp</title> 
</head> 
<body bgcolor= "cyan" > 
<% 
String strName = request. getParameter(" text") ; 
String strPassword = request. getParameter(" pwd") ; 
if (!("sa".equals(strName) || "admin123". equals(strPassword))) í 
response.sendRedirect("eg4 4.jsp"); // 重 定向 到 eg4_4.jsp 页 面 重新 输入 密码 
) else | 
out. print(" 管 理 员 登 录 成 功 ,您 可 以 管理 页 面 了 !"); 
) 
V» 
</body> 
</html> 


首先 运行 eg4_4. jsp 页 面 , 效 果 如 图 4-11 


http://localhost:8080/ch04/eg4_4jsp 


当 用 户 在 该 页 面 中 输入 正确 的 管理 员 账 号 


所 示 的 页 面 效果 。 如 果 账 号 和 密码 输入 错误 ， Wan 例子 4_4 页 面 效 果 
那么 还 会 重新 定向 到 最 初 页 面 。 


http://localhost:8080/ch04/eg4_4jsp 


http://localhost:8080/ch04/verifyjsp 


图 4-12 登录 页 面 效果 图 4-13 验证 页 面 效果 
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需要 注意 的 是 ,在 客户 端 浏览 器 工作 时 ,Web 服务 器 要 求 浏览 器 重新 发 送 一 个 请 求 ,用 
于 重 定向 某 个 页 面 , 在 浏览 器 地 址 栏 上 会 出 现 重 定向 页 面 的 URL, 且 为 绝对 路 径 。 

在 第 3 章 学 习 过 的 forward 动作 标记 也 可 以 实现 页 面 的 跳 转 ,例如 下 面 的 语句 : —jsp: 
forward page = " verify. jsp"/>. fH ffi H] forward 动作 标记 与 response 对 象 调用 
sendRedirect 方法 有 所 不 同 ,具体 区 别 分 析 如 下 : 

(1) forward 为 服务 器 端 跳 转 , 浏 览 器 地 址 栏 不 变 ; sendRedirect 为 客户 端 跳 转 ,浏览 器 
地 址 栏 改变 为 新 页 面 的 URL. 

(2) 执行 到 forward 标记 出 现 处 停止 当前 JSP 页 面 的 继续 执行 ,而 转向 标记 中 page Jš 
性 指定 的 页 面 ; sendRedirect 所 有 代码 执行 完毕 之 后 再 跳 转 。 

(3) 使 用 forward ,通过 request 请 求 信 息 能 够 保留 在 下 一 个 页 面 ; 使 用 sendRedirect 
不 能 保留 request 请 求 信 息 。 

(4) forward 传递 参数 的 格式 如 下 : 

<jsp:forward page= "verify.jsp"> 

<jsp:param name= "text" value= "sa" /2> 


<jsp:param name="pwd" value= "admin123"/> 
< /jsp:forward> 


response 对 象 的 sendRedirect 传递 参数 的 方式 如 下 : 


response. sendRedirect( " verify. jsp?text— sa&pwd-— admin123") ; 


4.2.3 拓展 训练 


1. 有 如 下 程序 段 : 
<form> 
<input type = "text" name = "id"> 
<input type= "submit" value 一" 提交 "过 
</form> 
下 面 ( ) 语 句 可 以 获取 用 户 输入 的 信息 。 
A. request. getParameter("id") ; B. request. getAttribute(" submit") ; 
C. session. getParameter(key. "id"); D. session. getAttribute (key. "id"); 
2. 下 面 ( ) 内 置 对 象 是 对 客户 的 请 求 做 出 响应 ,向 客户 端 发 送 数据 的 。 
A. request B. session C. response D. application 
3. 可 以 使 用 ( ) 方 法 实现 客户 的 重 定向 。 
A. response. setŠtatus(); B. response. setHeaderO ; 
C. response. setContent TypeO ; D. response. sendRedirectO) ; 


4. 什么 对 象 是 内 置 对 象 ? 常见 的 内 置 对 象 有 哪些 ? 
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5. 简 述 forward 动作 标记 与 response. sendRedirect() 两 种 跳 转 的 区 别 。 

6. 修改 4.2. 2 节 中 的 例题 eg4_3.jsp, 使 之 具备 如 下 功能 : 在 当前 页 面 停留 5 秒 钟 后 自 
动 跳 转 到 另 一 个 页 面 next. jsp(eg4_3. jsp 与 next. jsp 在 同一 个 Web 服务 目录 中 )。 提 示 : 
需要 为 eg4_3.jsp 设置 一 个 响应 头 , 也 就 是 在 eg4 3. jsp 页 面 中 添加 如 下 代码 : 


response. set Header( "refresh" , "5 ; url= next. jsp") ; 


7. 编写 两 个 JSP 页 面 input. jsp 和 compute. jsp, 用 户 可 以 使 用 input. jsp 页 面 提供 的 
表单 输入 两 个 运算 数 ,选择 一 种 运算 符 ,并 提交 给 JSP 页 面 compute. jsp 来 完成 相应 的 计算 
功能 ; 如 果 用 户 某 一 项 没有 输入 或 者 没有 选择 ,都 将 重新 定向 到 input. jsp 页 面 等 待 用 户 输 
A. input. jsp 运行 后 页 面 效 果 如 图 4-14 所 示 , 当 用 户 输入 运算 数 . 选择 运算 符 后 ， 
compute. jsp 页 面 计算 结果 如 图 4-15 所 示 。 


http://localhost:8080/chO4/inputjsp. 


NAN E 


图 4-14 输入 信息 图 4-15 显示 计算 结果 


代码 模板 input. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBI 
"GBK"%> 
<html> 
<head> 
<title>input.jsp</title> 
</head> 
<body bgcolor= "cyan" 
【代码 段 INV E X A hE 
本 网 页 用 于 进行 加 减 乘除 运算 . 
<p> 
请 输入 两 个 操作 数 ,并 选择 操作 符 : 
<p> 
<input type="text" name— "numberl"> 


pageEncoding — 


<select size— "1" name— "list" 

<option value 一 "" 二 操作 符 

【代码 段 23/ /在 下 拉 列 表 中 添加 “十 "、“ 一 ”"、“ x* ”、“/” 四 种 运算 符 
</select> 
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< input type="text" name= "number2"> 
«p 
<input type="submit" value— "JF fit $E" name— "button" 
</form> 
</body> 
</html> 


代码 模板 compute. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 


"GBK"%> 

<html> 

<head> 

<title> compute. jsp</title> 

</head> 

<body bgcolor= "cyan"> 

<% 
String numl = request. getParameter("numberl") ; 
String num2 = request. getParameter( " number2") ; 
String operator — request. getParameter("list") ; 
if(numl. length) 

【代码 段 3]// 重 定向 到 input. jsp 页 面 重新 输入 密码 


) 


else if(num1. length()!=0 &.&. num2.length()! —0 &.&. operator. length() ! =0) í 


ifCoperator. equals( "add")) 


out. print(numl +" +" + num2 +" =" + (Double. parseDouble( num1) + Double 


(num2))); 
else if(operator. equals(" sub") ) 


out. print(numl +" — " + num2 + " =" + (Double. parseDouble( numl1 ) — Double 


(num2))); 
else if(operator. equals( " mul") ) 


out. print(numl +" * " +-num2 + " =" + (Double. parseDouble( num1) * Double 


(num2))); 
else if(operator. equals("div")) 


out. print (num1 + "/" + num2 +" =" + (Double. parseDouble ( numl ) /Double. 


(num2))); 
) 

“> 

</body> 

</html> 


4.3 会 话 对 象 session 


一 0 || num2.length() — —0 || operator.length() 一 一 0){ 


. parseDouble 


. parseDouble 


. parseDouble 


. parseDouble 


客户 端 浏览 器 与 Web 服务 器 之 间 使 用 Http 协议 进行 通信 。Http 是 一 种 无 状态 协议 ， 
客户 向 服务 器 发 出 请 求 (request) ,服务 器 返回 响应 (response) ,连接 就 被 关闭 了 ,在 服务 器 
端 不 保留 连接 的 相关 信息 。 所 以 服务 器 必须 采取 某 种 手段 来 记录 每 个 客户 的 连接 信息 。 
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Web 服务 器 可 以 使 用 内 置 对 象 session 来 存放 有 关连 接 的 信息 。session 对 象 指 的 是 客户 端 
与 服务 器 端的 一 次 会 话 ,从 客户 端 连 到 服务 器 的 一 个 Web 应 用 程序 开始 ,直到 客户 端 与 服 
务 器 断 开 为 止 。 


4.3.1 知识 要 点 


1. session 对 象 的 ID 
Web 服务 器 会 给 每 一 个 用 户 自 动 创建 一 个 session 对 象 ,为 每 个 session 对 象 分 配 一 个 
唯一 标识 的 String 类 型 的 session ID, 这 个 ID 用 于 区 分 其 他 用 户 。 这 样 每 个 用 户 都 对 应 着 
一 个 session 对 象 (该 用 户 的 会 话 ) ,不 同 用 户 的 session 对 象 互 不 相同 。session 对 象 调用 
getId() 方 法 就 可 以 获取 当前 session 对 象 的 ID. 
2. session 对 象 存储 数据 
使 用 session 对 象 可 以 保存 用 户 在 访问 某 个 Web 服务 目录 期 间 的 有 关 数 据 。 处 理 数据 
的 方法 如 下 : 
* public void setAttribute(String key. Object obj? 
将 参数 obj 指定 的 对 象 保存 到 session 对 象 中 ,key 为 所 保存 的 对 象 指定 一 个 关键 字 。 
若 保存 的 两 个 对 象 关 键 字 相同 , 则 先 保存 的 对 象 被 清除 。 
* public Object getAttibute(String key) 
获取 session 中 关键 字 是 key 的 对 象 。 
* public void removeAttribute(String key) 
从 session 中 删除 关键 字 key 所 对 应 的 对 象 。 
* public Enumeration getAttributeNames() 
产生 一 个 枚 举 对 象 ,该 枚 举 对 象 可 使 用 方法 nextElemets O38 Jj. session 中 的 各 个 对 象 
所 对 应 的 关键 字 。 
3. session 对 象 的 生存 期 限 
一 个 用 户 在 某 个 Web 服务 目录 中 的 session 对 象 的 生存 期 限 依赖 于 以 下 几 个 因素 : 
* 用 户 是 否 关闭 浏览 器 。 
° session 对 象 是 否 调用 invalidate() 方 法 。 
* session 对 象 是 否 达到 设置 的 最 长 "发 呆 ? 时 间 。 
4.3.2 技能 操作 
具体 任务 如 下 : 
。 获取 session 对 象 的 生存 期 限 ID. 


。 获取 使 用 session 对 象 存储 数据 。 
° 理解 session 对 象 的 生存 期 限 。 
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1. 获取 session 对 象 的 ID 

编写 三 个 JSP Jt ifii eg4 5. jsp( 个 人 主页 ) information. jsp( 自 我 介绍 ) 和 major. jsp( 专 
业 介绍 ) ,存放 在 同一 Web 服务 目录 中 。 客 户 首先 访问 eg4_5. jsp 页 面 ,运行 结果 如 
图 4-16 所 示 , 从 该 页 面 可 以 链接 到 information. jsp 页 面 ,然后 从 information. jsp 页 面 还 可 
以 链接 到 major. jsp 页 面 。 


http://localhost:8080/ch04/eg4_5.jsp 


图 4-16 个 人 主页 
代码 模板 eg4_5. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK" A42 
<html> 
<head> 
<title>eg4_5.jsp</title> 
</head> 
<body bgcolor= "cyan" 
欢迎 访问 我 的 个 人 主页 ! 
<p> 
首先 了 解 一 下 Web 服务 器 为 我 分 配 的 session 对 象 的 ID: 
<% 
String id= session. getld() ; // 使 用 session 对 象 调用 getId 方法 获得 ID 
%> 
<br> 
<%=id X> 
<p> 
点 击 链接 去 <<a href= "information.jsp">> A MINA —/a7 A B? 
</body> 
</html> 


代码 模板 information. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 

<html> 

<head> 

<title>information. jsp</title> 

</head> 
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<body bgcolor= "cyan" > 
谢谢 您 关注 我 的 二 font size=5> B E fr ZH < / font ! br — br 
还 是 要 先 来 了 解 一 下 Web 服务 器 为 我 分 配 的 session 对 象 的 ID: 
<% 
String id= session. getId() ; // 使 用 session 对 象 调用 getId 方法 获得 ID 
%> 
<br> 
«Hid %> 
<br><br> 
我 生 在 一 个 小 山村 
那里 有 我 的 父老 乡亲 
胡子 里 长 满 故 事 
mp pH £4 
<br><br> 
点 击 链接 去 二 a href="major.jsp"> l 4r #4</a> RAE? 
</body> 
</html> 


运行 结果 如 图 4-17 所 示 。 


http://localhost:8080/ch04/information.jsp 


4-17 自我 介绍 
代码 模板 major. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title>major.jsp</title> 
</head> 
<body bgcolor= "cyan"> 
谢谢 您 关注 我 的 <font size=5> Jr #4</tont2>!<br><br> 
最 后 还 是 要 看 看 Web 服务 器 为 我 分 配 的 session 对 象 的 ID: 
<% 
String id=session. getId(); // 使 用 session 对 象 调用 getId 方法 获得 ID 
> 
<br> 
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<%=id ⁄> 
<br><br> 
<h> HAIE My </h22> 
本 专业 是 计算 机 硬件 与 软件 相 结合 、 面 向 系统 ,侧重 应 用 的 宽 口 径 专 业 。<=br 二 
通过 基础 教学 与 专业 训练 ,培养 基础 知识 扎实 、 知 识 面 宽 工程 实践 能 力 强 ,二 br 二 
具有 开拓 创新 意识 ,在 计算 机 科学 与 技术 领域 从 事 科 学 研究 教育 、 开 发 和 应 用 的 高 级 人 才 。 
<br> 
本 专业 开设 的 主要 课程 有 : 程序 设计 、 数 据 结构 、 操 作 系统 .计算 机 网 络 Pis SR EAR. <br> 
<br><br> 
点 击 链接 去 一 a href= "eg4_5.jsp" 过 我 的 首页 一 /a> 看 看 吧 ? 
</body> 
</html> 


运行 结果 如 图 4-18 所 示 。 


| http://localhost:8080/ch04/major.jsp 


图 4-18 专业 介绍 


从 以 上 三 个 页 面 的 运行 结果 来 看 ,一 个 用 户 在 同一 个 Web 服务 目录 中 只 有 一 个 
session 对 象 , 当 用 户 访问 相同 Web 服务 目录 的 其 他 页 面 时 , Web 服务 器 不 会 再 重新 分 配 
session 对 象 ,直到 用 户 关闭 浏览 器 或 这 个 session 对 象 达到 了 它 的 生存 期 限 。 当 用 户 重 新 
打开 浏览 器 再 访问 该 Web 服务 目录 时 , Web 服务 器 为 该 客户 再 创建 一 个 新 的 session 
对 象 。 

需要 注意 的 是 ,同一 用 户 在 多 个 不 同 的 Web 服务 目录 中 所 对 应 的 session 对 象 是 不 同 
的 ,一 个 服务 目录 对 应 一 个 session 对 象 。 

2. 使 用 session 对 象 存储 数据 

编写 三 个 JSP 页 面 eg4_6. jsp、display. jsp 和 feedback. jsp, 来 模拟 学 生 在 线 信息 填报 
系统 。 其 中 eg4_6. jsp 页 面 用 于 填写 个 人 资料 ,display. jsp 页 面 用 于 显示 已 填写 信息 ， 
feedback. jsp 页 面 用 于 为 已 确认 提交 个 人 资料 的 学 生 显 示 反 馈 信息 。 

代码 模板 eg4_6. jsp 如 下 : 
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<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title> eg4_6.jsp</title> 
</head> 
<body bgcolor= "cyan" > 
<form method="post" action= "display. jsp"> 
去 p> 学 生 个 人 资料 填写 一 br> 
text" name= "stuName" > #E $$: <input type="text" name= "age"> 
text" name= "birth" My : <input type— "text" name= "major" > 
p TESI: <input type="radio" value=" $ "ame— "sex" 8 
<input type="radio" value— "4t" name— "sex" Zr 
电话 : <input type="text" name— "phone" 
一 p> 爱 好 : <input type="checkbox" name= "hobby" value= "flit" lli 
<input type="checkbox" name= "hobby" value— " # #£ "> Fik 
<input type="checkbox" name= "hobby" value 一 "电影 "之 电影 
<input type="checkbox" name= "hobby" value=" E R "> E fj 
<input type="checkbox" name= "hobby" value 二 "游泳 "二 游泳 
<input type="checkbox" name= "hobby" value— "4T3R "1T 52K 
<p><input type="submit" value 二 "确定 " name— "yes" 
<input type— "reset" value— "3E v" name— "no" 
</form> 
</body> 
</html> 


代码 模板 display. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"% > 
<%@ page import ="java. util. * " % > 
<html> 
<head> 
<title>display. jsp</title> 
</head> 
<body bgcolor= "cyan"> 
<% 
request. setCharacterEncoding( "GBK" ) ; 
String studentName = request. getParameter( "stuName") ; 
// 下 面 的 语句 是 将 学 生 姓 名 studentName 以 “studentName” 为 关键 字 存 储 到 session 对 象 中 
session.setAttribute( "studentName", studentName) ; 
String stuAge = request. getParameter( "age" ) ; 
session. setAttribute("stuAge", stuAge) ; 
String stuBirth = request. getParameter( " birth") ; 
session. setAttribute("stuBirth", stuBirth) ; 
String stuMajor = request. getParameter( " major") ; 
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session. setAttribute("stuMajor", stuMajor) ; 
String stuSex — request. getParameter("sex") ; 
session. setAttribute("stuSex", stuSex) ; 
String stuPhone = request. getParameter(" phone") ; 
session. setAttribute("stuPhone", stuPhone) ; 
String stuHobby[] = request. getParameterValues(" hobby") ; 

if(stuHobby! = null) 

{ for(int k—0;k- stuHobby. length; k4- +) 

Í session. setAttribute(stuHobby[k] , stuHobby[k]) ;// 将 相应 的 关键 字 存 储 到 session 对 象 中 
) 


姓名 : <% =studentName% > 
年 龄 : <% =stuAge 47» — br 
籍贯 : <% =stuBirth 47 

专业 : <% — stuMajor 967 — br 
TESI: <% —stuSex 47 

电话 : <% —stuPhone 267 — br 


for(int i—0;i- stuHobby. length ;i-- +) ( 
out. print(stuHobby[i] +" "5; 

) 

6 
<p> 
<form action= "feedback. jsp" method=" post" > 

<input type="submit" value= "MU % t" /> 
</form> 
<a href= "eg4_6. jsp" > EHM S5 </a> 
</body> 
</html> 


代码 模板 feedback. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK" 4 
<html> 
<head> 
<title> feedback. jsp</title> 
</head> 
<body bgcolor= "cyan"> 
这 里 是 信息 反馈 页 面 ， 
<% 
// 下 面 的 语句 是 获取 session 中 关键 字 为 studentName 的 对 象 
String name= (String)session. getAttribute( "studentName"); 
out. print( name) ; 42» 
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的 信息 已 确认 提交 . 

p Sli BE XE a href 一 "eg4_6.jsp" 盖 该 例 首页 可 以 修改 信息 二 /a> 
</body> 
</html> 


运行 结果 如 图 4-19 所 示 。 


http://localhost:8080/ch04/eg4 6jsp 


计算 机 


123456789 


| 确定 | EZ | 


4-19 信息 填写 页 面 效果 


当 用 户 填 写 完 信息 ,并且 单 击 “确定 ?按钮 后 ,会 进入 如 图 4-20 所 示 的 信息 显示 页 面 。 
当 用 户 在 信息 展示 页 面 中 再 次 单 击 * 确 认 完毕 ?按钮 后 ,会 出 现 如 图 4-21 所 示 的 信息 反 
馈 页 面 。 


http://localhost:8080/ch04/displayjsp 


http;//localhost:8080/ch04/feedback jsp 


图 4-20 信息 展示 页 面 图 4-21 信息 反馈 页 面 


从 上 例 中 可 以 看 出 , 当 某 个 用 户 进 入 某 个 Web 服务 目录 ,服务 器 就 会 为 该 用 户 分 配 一 
个 session 对 象 ; 用 户 可 在 该 Web 服务 目录 的 所 有 页 面 使 用 这 个 session 对 象 ,该 session 
可 以 存储 ,获取 和 移 除 用 户 的 信息 对 象 ; 当 用 户 离开 该 Web 服务 目录 时 ,session 对 象 随 即 
也 会 消失 。 

3. session 对 象 的 生存 期 限 

编写 一 个 JSP 页 面 eg4_7. jsp, 完 成 下 述 功能 : 用 户 如 果 是 第 一 次 访问 该 页 面 ,向 用 户 
展示 欢迎 信息 ,并 输出 会 话 对 象 允 许 的 最 长 “发 果 ” 时 间 、 创 建 时 间 以 及 session 的 ID, fE 
eg4_7.jsp 页 面 中 ,session 对 象 使 用 setMaxlnactivelnterval(int maxValue) 方 法 设置 最 长 的 
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“发 果 ” 状 态 时 间 为 8 秒 。 用 户 如 果 两 次 刷新 间隔 时 间 超 过 8 秒 , 用 户 先 前 的 session 被 取 
消 , 用 户 将 获得 一 个 新 的 session 对 象 。 
代码 模板 eg4_7.jsp 如 下 : 


<% @ page language =" java" contentType = " text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<%@ page import= "java. util. * "% > 
<%@ page import= "java. text. * "% > 
<html> 
<head> 
<meta http-equiv= "Content-Type" content= "text/html; charset= ISO-8859-1"> 
<title>eg4_7.jsp</title> 
</head> 
<body> 
<% 
// 下 面 语句 的 功能 : session 调用 setMaxInactivelnterval(int n) 方 法 设置 最 长 “发 果 ” 时 间 为 10 秒 
session. setMaxlInactiveInterval(8) ; 
boolean oper = session. isNew() ; //session 调用 isNew() 方 法 判断 session 是 不 是 新 创建 的 
if (oper) { 
out. println(" 欢 迎 您 首次 访问 当前 Web 服务 目录 ch04."); 
out. println("<hr/>"); 
} 
out. println( "session 允许 的 最 长 发 呆 时 间 为 : " 十 
session. getMaxJInactiveInterval() 十 " 秒 ."); 
// 获 取 session 对 象 被 创建 的 时 间 
long num = session. getCreationTime() ; 
// 将 整数 转换 为 Date 对 象 
Date time = new Date(num); 
// 用 给 定 的 模式 和 默认 语言 环境 的 日 期 格式 符号 构造 SimpleDateFormat 对 象 
SimpleDateFormat matter 一 new SimpleDateFormat( 
"北京 时 间 : yyyy 年 MM 月 dd 日 HH 时 mm 分 ss 秒 E."); 
// 得 到 格式 化 后 的 字符 串 
String strTime = matter. format(time) ; 
out. println(" — br/7 session 的 创建 时 间 为 : " + strTime) ; 
out. println("  br/2» session 的 id 为 : " + session.getld() + "."); 
5 
</body> 
</html> 


第 一 次 访问 eg4_7.jsp 页 面 ,会 出 现 如 图 4-22 所 示 的 页 面 效果 。 如 果 在 8 秒 钟 之 内 刷 
新 该 页 面 , 即 在 8 秒 钟 之 内 再 次 访问 该 页 面 ,会 出 现 如 图 4-23 所 示 的 页 面 效果 ; 如 果 超 过 8 
秒 钟 之 后 再 刷新 该 页 面 , 即 在 8 秒 钟 之 后 再 次 访问 该 页 面 , 仍 会 出 现 类 似 图 4-22 所 示 的 效 
果 , 但 页 面 中 session 的 id 值 将 与 上 次 出 现 的 id 值 不 同 。 
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http://localhost:8080/ch04/eg4_7jsp 


图 4-22. 首次 访问 或 间隔 10 秒 后 访问 该 页 面 


http://localhost:8080/ch04/eg4 7 jsp 


图 4-23 10 秒 之 内 访问 该 页 面 


从 上 例 中 可 以 看 出 ,如 果 用 户 长 时 间 不 关闭 浏览 器 ,session 对 象 也 没有 调用 invalidate 
() 方 法 ,那么 用 户 的 session 也 可 能 消失 。 例 如 eg4 7. jsp 页 面 在 8 秒 之 内 不 被 访问 的 话 ， 
它 先 前 创建 的 session 对 象 就 消失 了 ,服务 器 又 重新 创建 一 个 session 对 象 。 这 是 因为 
session 对 象 达到 了 它 的 最 大 “发 果 ” 时 间 。 所 谓 “ 发 果 ” 状 态 时 间 , 是 指 用 户 对 某 个 Web JR 
务 目 录 发 出 的 两 次 请 求 之 间 的 间隔 时 间 。 

用 户 对 某 个 Web 服务 目录 下 的 JSP 页 面 发 出 请 求 并 得 到 响应 ,如 果 用 户 不 再 对 该 
Web 服务 目录 发 出 请 求 ,例如 不 再 操作 浏览 器 ,那么 用 户 对 该 Web 服务 目录 进入 “发 果 ” 状 
态 , 直 到 用 户 再 次 请 求 该 Web 服务 目录 时 性 发呆” 状态 结束 。 

Tomcat 服务 器 允许 用 户 最 长 的 “发 果 ” 状 态 时 间 为 30 分 钟 。 可 以 通过 修改 Tomcat 安 
装 目 录 中 conf 文件 夹 下 的 配置 文件 web. xml 文件 ,找到 下 面 的 片断 ,修改 其 中 的 默认 值 
30, 就 可 以 重新 设置 各 个 Web 服务 目录 下 的 session 对 象 的 最 长 发呆” 时间 。 这 里 的 时 间 
单位 为 分 钟 。 


<session-config> 
<session-timeout>30</session-timeout> 
</session-config> 
也 可 以 通过 session 对 象 调用 setMaxlnactivelnterval(int time) 方 法 来 设置 最 长 “发 果 ” 状 态 
时 间 ,参数 的 时 间 单 位 为 秒 。 


4.3.3 拓展 训练 


1. 一 个 用 户 在 不 同 Web 服务 目录 中 的 session 对 象 相 同 吗 ? 一 个 用 户 在 同一 Web 服 
务 目录 的 不 同 子 目录 中 的 session 对 象 相同 吗 ? 
2. session 对 象 的 生存 期 限 依赖 于 哪些 因素 ? 
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3. 编程 实现 下 面 的 功能 : 用 户 到 社区 便利 店 选 购 商品 ,购物 前 需 先 登录 会 员 卡 号 , 购 
买 时 先 把 选 购 的 商品 放 和 购物 车 ,然后 到 柜台 清点 付款 。 请 借助 session 对 象 相关 知识 来 模 
拟 网 上 购物 车 ,并 存储 客户 的 会 员 卡 号 以 及 所 购买 的 商品 名 称 。 编 写 程序 模拟 上 述 过 程 : 
login. jsp 实现 会 员 卡 号 输入 ,lead. jsp 实现 商品 导购 ,buy. jsp 实现 商品 购买 ,list. jsp 实现 
商品 清点 。 首 先 运行 login. jsp ,页 面 效果 如 图 4-24 所 示 ; 当 输 入 会 员 卡 号 并 单 击 “ 登 录 ” 按 
钮 后 ,会 员 卡 号 被 送出 ,此 时 继续 在 该 页 单 击 链接 ,会 进入 lead. jsp 商品 导购 页 面 , 如 图 4-25 
所 示 ; 在 lead. jsp 商品 导购 页 面 中 ,可 以 单 击 “ 欢 迎 选 购 所 需要 的 文具 ”链接 ,进入 buy. jsp 
商品 购物 页 面 ,如 图 4-26 所 示 ; 在 lead. jsp 商品 导购 页 面 中 ,用 户 可 以 勾 选 所 购买 的 商品 ， 
然后 单 击 “ 确 定 ” 按 钮 ,进入 最 终 的 商品 清点 页 面 ,如 图 4-27 所 示 。 要 求 将 如 下 代码 模板 补 
充 完整 ,并 调试 运行 。 


httpWlocalhost8080/cho4/loginjsp 


Et 


图 4-24 会 员 登录 页 面 图 4-25 商品 导购 页 面 


http://localhost:8080/ch04/buyjsp http://localhost:8080/ch04/listjsp 


图 4-26 商品 购买 页 面 图 4-27 商品 清点 页 面 


代码 模板 login. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title>login. jsp</title> 
</head> 
<body bgcolor= "cyan"> 
<form method="post" > 
欢迎 光临 糖 糖 社区 便利 店 ,请 输入 您 的 会 员 卡号 : 
<p> 
< input type="text" name= "userID"— 
<input type="submit" value 一 "登录 " name= "button" 
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</form> 
<% 
【代码 段 1]// 获 取 用 户 在 文本 框 userID 中 输入 的 信息 ,并 存储 到 字符 串 对 象 userID 中 
if(userID= =null) 
{ userlD=" "; 
) 
else 
{ 
【代码 段 21//1 2 PA F userID 以 customerInfo 为 关键 字 存储 到 session XJ # f 
) 
%> 
<a href— "lead. jsp" > fci HE Ab BE n] DAWAH ia ! </a> 
</body> 
</html> 


代码 模板 lead. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title> lead. jsp</title> 
</head> 
<body bgcolor= "cyan" > 
这 里 是 文具 柜台 : 
<p> 
<a href= "buy.jsp">> XC il W jr $ë 3€ f X R... </a> 
<p> 
【代码 段 31// i EHE] login.jsp 页 面 
您 也 可 以 单 击 这 里 修改 会 员 卡 号 信息 .二 /a 二 
</body> 
</html> 


代码 模板 buy. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK" 4 
<html> 
<head> 
<title>buy.jsp< /title> 
</head> 
<body bgcolor= "cyan"> 
<form method="post" action— "list. jsp"> 
这 里 是 文具 柜台 ,请 选择 所 需 文具 
<br> 
<input type="checkbox" name= "hobby" value= "£ = £ "— 4r E 
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【代码 段 4]// 添 加 “ 直 尺 ”选项 
<input type="checkbox" name- "hobby" value= " # £ JJ ">> #ë 7] 
<input type="checkbox" name= "hobby" value 二 "透明 胶带 "二 透明 胶带 
<input type="checkbox" name= "hobby" value=" BA ">W PA 
<input type="checkbox" name= "hobby" value 
<p><input type="submit" value— "ff E" name 
<input type= "reset"value= "Æ E" name= "no" 
</form> 
</body> 
</html> 


代码 模板 list. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<%@ page import ="java. util. * " %> 
<html> 
<head> 
«title list. jsp /title> 
</head> 
<body bgcolor= "cyan"> 
<% 
request. setCharacterEncoding( " GBK") ; 
String foods[] = request. getParameterValues(" hobby") ; 
if (foods! = null) 
{ for(int k—0;k- foods. length; k-- +) 
(REE 5]/ / E foods[k] 相 应 的 关键 字 存 储 到 session 对 象 中 
} 
} 
%> 
这 里 是 收银 台 , 请 确认 以 下 购买 信息 : 
<p> 
会 员 卡 号 : 
<% 
【代码 段 6]// 获 取 session 中 关键 字 为 customerInfo 的 对 象 ( 会 员 卡号 ) , 存 人 字符 串 对 象 ID 中 
out. println(ID) ; 
> 
<br> 
所 购 商 品 清单 : 
<% 
for(int i=0;i< foods. length;i+ +) ( 
out. print( foods [i] +" "); 
) 
> 
</body> 
</html> 
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4.4 application 对 象 


4.4.1 知识 要 点 


不 同 用 户 的 session 对 象 互 不 相同 ,但 有 时 候 用 户 之 间 可 能 需要 共享 一 个 对 象 , Web 服 
务 器 启动 后 ,就 产生 了 这 样 一 个 唯一 的 内 置 对 象 , 即 application 全 局 应 用 程序 对 象 。 任 何 
用 户 在 访问 同一 Web 服务 目录 的 各 个 页 面 时 ,共享 一 个 application 对 象 ,直到 服务 器 关闭 ， 
这 个 application 对 象 被 取消 为 止 。application 同 session 对 象 一 样 也 可 以 进行 数据 的 存储 ， 
处 理 数据 的 方法 如 下 : 

* public void setAttribute(String key, Object obj) 

将 参数 obj 指定 的 对 象 保存 到 application 对 象 中 ,key 为 所 保存 的 对 象 指 定 一 个 关键 
字 。 若 保存 的 两 个 对 象 关 键 字 相同 , 则 先 保存 的 对 象 被 清除 。 

* public Object getAttribute(String key) 

获取 application 中 关键 字 是 key 的 对 象 。 

* public void removeAttribute(String key) 

从 application 中 删除 关键 字 key 所 对 应 的 对 象 。 

* public Enumeration getAttributeNames() 

产生 一 个 枚 举 对 象 ,该 枚 举 对 象 可 使 用 方法 nextElemets O38 BJ application 中 的 各 个 
对 象 所 对 应 的 关键 字 。 


4.4.2 技能 操作 


使 用 application 对 象 存储 数据 的 功能 制作 简单 的 网 络 聊天 室 。 具 体 任 务 描述 如 下 : 编 
写 三 个 JSP 页 面 文件 eg4_8_1.jsp、eg4_8_2.jsp 和 eg4_8_3.jsp。 其 中 eg4_8_1. jsp 为 聊天 
室 的 首页 ,用户 可 以 在 该 页 输入 昵称 和 聊天 内 容 , 同 时 可 将 发 言 内 容 提 交 给 eg4_8_2. jsp 页 
Hü; eg4_8_2. jsp 页 面 负责 获取 用 户 输入 的 信息 并 存储 起 来 ; eg4_8_3. jsp 负责 显示 所 有 用 
户 的 聊天 内 容 。 

代码 模板 eg4 8 l.jsp 如 下 : 


«(3 page contentType— "text/html;charset— gb2312" % > 

<html> 

<body bgcolor= "cyan" > 

<form action="eg4_8_2.jsp" method="post" name= "form"> 

欢迎 来 到 简 简单 单 聊 天 室 ,畅所欲言 ! 
过 br 二 上 昵称: <input type="text" name= "peopleName"> 
去 br> 输 入 您 的 聊天 内 容 : 
<br><textarea name= "messages" rows— "10" cols—36 wrap= "physical" >< /textarea2> 
«br input type="submit" value 一 "提交 发 言 " name= "submit" 
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</form> 
<form action 一 "eg4_8_3.jsp" method="post" name= "form1">> 
<input type="submit" value 二 "查看 聊天 记录 " name= "look”> 
</form> 
</body> 
</html> 


代码 模板 eg4_8_2. jsp 如 下 : 


<%@ page contentType= "text/html;charset— gb2312" pageEncoding— "GBK" % > 
<%@ page import— "java. util. * " 267» 
«html <body bgcolor— "cyan" 
«4! Vector v—new Vector() ; 
int i—0; ServletContext application; 
synchronized void sendMessage(String s) 
{ application getServletContext( ) ; ; 
vta 
v. add("No. "it "M" s); 
// 下 面 语句 的 功能 是 把 聊天 内 容 v DA" Mess" HARFE] application XJ $ rh 
application. setAttribute(" Mess", v) ; 
) 
> 
<% request. setCharacterEncoding( " GBK") ; 
String name= request. getParameter( " peopleName") ; 
String messages = request. getParameter(" messages") ; 
if(name— — null) 
{name= "guest" + (int) ( Math. random() * 10000) ; 
) 
if(messages= = null) 
{messages 一 "无 信息 "; 
) 
String s 一 "昵称 :" 十 name 十 "# " + "MAWA :" + "< BR2>" + messages: 
sendMessage(s) ; 
out. print(" 您 的 发 言 已 经 提交 !"); 
%> 
<a href= "eg4_8_1.jsp" >É [sl 
</body> 
</html> 


代码 模板 eg4_8_3. jsp 如 下 : 


<%@ page contentType= "text/html;charset=GB2312" %> 
<%@ page import= "java. util. * " %> 
<html>< body bgcolor= "cyan"> 
<% 
// 下 面 语句 的 功能 是 提取 application 中 关键 字 是 Mess 的 对 象 (聊天 内 容 ) 
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Vector v= (Vector)application. getAttribute(" Mess") ; 
for(int i=0;i<v. size() ;i+ +) 
{ String message= (String) v. elementAt(i) ; 
StringTokenizer fenxi= new StringTokenizer( message, " # "); 
while(fenxi. hasMoreTokens() ) 

( String str—fenxi. nextToken() ; 
byte a[] —str. getBytes( " GBK") ; 
str— new String(a) ; 

out. print( "<BR>" + str) ; 
) 


%> 
</body> 
</html> 
首先 运行 eg4_8_1. jsp. VE Ti CA IE] 4-28 所 示 ; 当 用 户 输入 昵称 和 聊天 内 容 , 单 击 “ 提 
交 发 言 ?按钮 后 ,会 进入 如 图 4-29 所 示 的 提示 页 面 ; 在 该 提示 页 面 中 单 击 “ 返 回 ? 超 链接 ,会 
再 次 回 到 首页 即 图 4-28 所 示 的 页 面 ,此 时 可 以 单 击 首 页 中 的 “查看 聊天 内 容 ” 按 钮 ,查看 聊 
天 内 容 , 如 图 4-30 所 示 。 


http://localhost:8080/ch04/eg4_8_1jsp 


的 一 种 表现 形式 ， 是 通过 互联 网 传播 的 
Fo HERE ` 焦点 问题 


MADARIA V BERT C 

观点 。 

“网络 组 情 是 以 网 络 为 载体 ， 以 事件 为 

Si. eius DE Pn om rr re 
图 4-28 聊天 室 首 页 图 4-29 ”提示 页 面 


http://localhost:8080/ch04/eg4_8_3.jsp 


图 4-30 查看 聊天 记录 页 面 
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上 例 eg4 8. 2. jsp 文件 中 的 sendMessage 方法 之 所 以 定义 为 同步 方法 ,是 因为 
application 对 象 针 对 所 有 的 用 户 都 是 同等 的 ,任何 用 户 对 该 对 象 中 存储 的 数据 的 操作 都 会 
影响 到 其 他 用 户 。 如 果 客 户 浏览 不 同 的 Web 服务 目录 ,将 产生 不 同 的 application 对 象 。 同 
一 个 Web 服务 目录 中 的 所 有 JSP 页 面 都 共享 同一 个 application 对 象 ,即使 浏览 这 些 JSP 
页 面 的 是 不 同 的 用 户 也 是 如 此 。 因 此 ,保存 在 application 对 象 中 的 数据 不 仅 可 以 跨 页 面 分 
享 , 还 可 以 由 所 有 用 户 共享 。 

有 些 Web 服务 器 不 能 直接 使 用 application 对 象 ,必须 使 用 父 类 ServletContext 声明 这 
个 对 象 ,然后 使 用 getServletContext() 方 法 为 application 对 象 进行 实例 化 。 例 如 该 任务 中 
eg4_8_2. jsp 页 面 中 的 代码 。 


4.4.3 拓展 训练 


1. 请 简 述 内 置 对 象 request、session 和 application 之 间 的 区 别 。 
2. 模仿 4.4.2 节 中 的 例题 ,使 用 application 对 象 实现 网 站 访客 计数 器 的 功能 。 访 客 计 
数 器 运行 页 面 效果 如 图 4-31 所 示 。 要 求 将 代码 模板 补充 完整 ,并 调试 运行 。 


http://localhost:8080/ch04/ 


图 4-31 访客 计数 器 


代码 模板 如 下 : 


<%G@ page contentType= "text/html;charset— gb2312" pageEncoding= "GBK" % > 
<html> 
<body bgcolor= "cyan" > 
<% 
【代码 段 11// BE application 中 关键 字 是 count 的 对 象 (计数 内 容 ) 
if(count= =null) ( 
count="1"; 
}else{ 
count= Integer. parseInt(count) 41+ ""; 
) 
【代码 段 2]// 把 计数 内 容 count 以 关键 字 “count" 为 关键 字 存储 到 application 对 象 中 


“> 
<N="<hi fiti NAK AN, RER" +H count+ f S UE E </h12><br>"⁄> 
</body> 


</html> 
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4.5 小 结 


所 有 的 内 置 对 象 不 需要 由 JSP 的 编写 者 声明 和 实例 化 ,可 以 直接 在 所 有 的 JSP 网 页 
中 使 用 。 内 置 对 象 只 在 Java 程序 片 或 者 Java 表达 式 中 使 用 。 

请 求 对 象 request 代表 客户 端 发 出 的 请 求 信息 对 象 , 通 常用 来 获取 表单 上 的 信息 。 
request 对 象 只 在 从 客户 发 出 请 求 到 服务 器 做 出 响应 这 个 期 间 是 有 效 的 。 

应 答对 象 response 代表 从 服务 器 端 返回 给 客户 端的 响应 信息 对 象 。response 对 象 
包含 从 动态 页 面 返回 给 客户 的 所 有 信息 。response 可 以 用 来 进行 页 面 的 重 定向 ,但 
与 用 forward 动作 标记 实现 页 面 的 跳 转 有 所 不 同 。 

会 话 对 象 session 可 以 用 来 保存 每 个 用 户 信息 ,以 便 跟 踪 每 个 用 户 的 操作 状态 ,不同 
JH P! fi session 对 象 互 不 相同 session 对 象 主要 用 来 存储 和 获取 数据 ,使 用 时 要 注 
意 其 生存 期 限 。 

全 局 应 用 程序 对 象 application 显示 相应 网 页 所 有 应 用 程序 的 对 象 。 任 何 用 户 在 访 
问 同一 Web 服务 目录 的 各 个 页 面 时 ,共享 一 个 application 对 象 ,直到 服务 器 关闭 ， 
这 个 application 对 象 被 取消 为 止 。 保 存在 application 对 象 中 的 数据 不 仅 可 以 跨 页 
面 分 享 ,还 可 以 由 所 有 用 户 共 享 。 


JavaBean 的 使 用 


通过 前 几 章 的 学 习 , 我 们 已 经 知道 一 个 JSP 页 面 通过 使 用 HTML 标记 为 用 户 显示 数 
据 ( 静 态 部 分 ) ,而 页 面 中 变量 的 声明 程序 片 以 及 表达 式 为 用 户 处 理 数据 (动态 部 分 )。 如 果 
Java 程序 片 和 HTML 标记 大 量 掺 杂 在 一 起 使 用 ,就 不 利于 JSP 页 面 的 扩展 和 维护 。JSP 
和 JavaBean 技术 的 结合 不 仅 可 以 实现 数据 的 表示 和 处 理 分 离 , 而 且 可 以 提高 JSP 程序 代码 
重用 的 程度 ,是 JSP 编程 中 常用 的 技术 。 

在 本 章 中 ,我 们 新 建 一 个 Web 工程 ch05 ,本 章 例子 中 涉及 的 Java 源 文件 保存 在 ch05 
的 src 中 ,涉及 的 JSP 页 面 保存 在 ch05 的 WebContent 中 。 


5.1 编写 JavaBean 


5.1.1 知识 要 点 


JavaBean 是 一 个 可 重复 使 用 的 软件 组 件 ,是 遵循 一 定 标准 .用 Java 语言 编写 的 一 个 类 ， 
该 类 的 一 个 实例 称 为 一 个 JavaBean, HEK bean, 

由 于 JavaBean 是 基于 Java 语言 的 ,因此 JavaBean 具有 以 下 特点 : 

。 与 平台 无 关 。 

。 代码 的 重复 利用 。 

* BA EHE . 易 使 用 。 

编写 一 个 JavaBean 就 是 编写 一 个 Java 的 类 (该 类 必须 带 有 包 名 ) ,这 个 类 创建 的 一 个 
对 象 称 为 一 个 bean, 为 了 让 JSP 引擎 (例如 Tomcat) 知 道 这 个 bean 的 属性 和 方法 ,必须 在 
类 的 方法 命名 上 遵守 以 下 规则 ， 

(1) 如 果 类 的 成 员 变 量 的 名 字 是 name, 那 么 为 了 获取 或 更 改 成 员 变 量 的 值 ,类 中 必须 
提供 两 个 方法 : 

* getName() 一 一 用 来 获取 属性 name, 

。 setName() 一 一 用 来 设置 属性 name。 

即 方法 的 名 字 用 get 或 set 为 前 级 ,后 级 是 首 字母 大 写 的 成 员 变 量 的 名 字 。 

(2) 对 于 boolean 类 型 的 成 员 变 量 ,允许 使 用 “is” 代 苦 上 面 的 “get” 和 “set”。 

(3) 类 中 声明 的 方法 的 访问 权限 都 必须 是 public. 

(4) 类 中 声明 的 构造 方法 必须 是 public、 无 参数 。 
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5.1.2 技能 操作 


使 用 JavaBean 的 编写 规则 编写 创建 bean 的 Java 源 文 件 。 具 体 任务 如 下 : 

* 在 Eclipse 中 创建 一 个 类 Student, 属 于 china. dalian 包 。 

。 编写 bean 的 源 文 件 Student. java( 在 包 china. dalian 中 ), 该 bean 的 作用 是 求学 生 

数学 课 和 外 语 课 的 总 分 。 

1. 在 Eclipse 中 创建 Student 类 

在 Eclipse 中 ,右键 单 击 ch05 项 目 , 然 后 在 弹出 的 快捷 菜单 中 选择 New-— Class 命令 ， 
会 弹出 新 建 类 对 话 框 , 在 该 对 话 框 中 可 以 设置 包 名 和 类 名 ,如 图 5-1 所 示 。 设 置 完 以 后 的 目 
录 结 构 如 图 5-2 所 示 。 


Java Class 


Create a new Java class. 


Source folder ch05/src | Browse. | 
| Package: V china dakan ] x Browse. _ 
E] Enclosing type: || Browse.. 

Name: Student 
Modifiers: @ public @ default private protected 
abstract [F] final [D static 
Superclass: java.lang.Object | Browse. | 
Interfaces: Add... 
Remove 


Which method stubs would you like to create? 

public static void main(String[] args) 

Constructors from superclass 

[V] Inherited abstract methods 

Do you want to add comments? (Configure templates and default value here) 
[-] Generate comments 
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= ch05 
4 @5 src 

4 8B china.dalian 
b [A] Studentjava 


图 5-2 目录 结构 


2. 编写 bean 的 源 文件 Student. java 
代码 模板 Student. java 如 下 : 


package china. dalian; 
public class Student{ 
int id; 
String name; 


double math, english; 


public Student() | // 定 义学 生 类 的 构造 方法 


id=1; 
name= "superMan"; 
math=100.0; 
english=100.0; 

) 


public int getIDO // 定 义 获 取 学 生 学 号 的 方法 
return id; 

) 

public void setID(int id) ( // 定 义 修改 学 生 学 号 的 方法 
this.id = id; 


) 

public String getName() { 
return name; 

) 

public void setName(String name) | 
this. name — name; 

) 

public double getMath() ( 
return math; 

) 

public void setMath(double math) ( 
this. math = math; 

) 

public double getEnglish() { 
return english; 

) 

public void setEnglish(double english) ( 
this.english — english; 
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) 
public double getSum() ( 
return math english; 
) 
) 


JavaBean 可 以 在 任何 Java 程序 编辑 环境 下 完成 编写 ,再 通过 编译 生成 一 个 字 节 码 文件 
C class 文件 ) ,为 了 让 JSP 引擎 (例如 Tomcat) 找 到 这 个 字 节 码 , 必 须 把 字 节 码 文件 放 在 特 
定 的 位 置 。 本 书 使 用 Eclipse 集成 环境 开发 JSP 程序 ,Java 类 的 字 节 码 文件 由 Eclipse 自动 
保存 到 Web 工程 的 build\classes 中 。 例 如 ,该 任务 中 的 Student. class 文件 保存 在 ch05N 
build\classes\china\dalian 目录 中 。 


5.1.3 拓展 训练 
1. JavaBean 中 声明 的 方法 的 访问 属性 必须 是 ( — D. 


A. private B. public C. protected D. friendly 
2, 写 一 个 bean 时 ,与 布尔 逻辑 类 型 的 成 员 变量 xxx 对 应 的 方法 是 ( Jis 
A. getXxx() B. setXxxO C. XxxO D. isXxxO 


3. 创建 bean 的 源 文件 Rect. java fE $D) graph. picture 中 ) ,该 bean 的 作用 是 计算 矩形 
的 周 长 和 面积 。 要 求 将 下 面 的 代码 模板 补充 完整 ,并 进行 调试 。 
代码 模板 Rect. java 如 下 : 


package graph. picture; 
public class Rect ( 
double length; 
double width; 


REE 1] // 定 义 类 Rect 的 构造 方法 
length=20. 56; 
width=15.45; 

) 

REE 2) // 定 义 获取 和 矩形 长 度 的 方法 


return length; 
) 
【代码 段 314 // 定 义 修改 矩形 长 度 的 方法 
this.length = length; 
) 
public double getWidth() { 
return width; 
) 
public void setWidth(double width) ( 
this. width — width; 


) 
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public double getGirth() ( 
return (length+- width) * 2; 
) 
public double getArea() í 
【代码 段 4]//3E EE JÉ BJ LA 
) 
) 


5.2 使 用 JavaBean 


5.2.1 知识 要 点 


在 JSP 页 面 中 使 用 bean, 首 先 必 须 使 用 page 指令 的 import 属性 导入 创建 bean 的 类 所 
在 的 包 , 例 如 : 

<%@ page import=" china. dalian. * "% > 
然后 使 用 JSP 动作 标记 useBean, 来 创建 与 使 用 bean, useBean 标记 的 格式 为 : 

<jsp:useBean id= "bean BJ" class= "@J#Ë bean 的 类 ”scope 一 "bean 的 有 效 范围 "/ 汪 
或 

去 jsp:useBean id= "bean 的 名 字 " class 一 "创建 bean 的 类 "scope 一 "bean [If IGI HE] 7 — / jsp: useBean 
例如 ， 

—jsp:useBean id= "rectangle" class= "small. dog. Rectangle" scope= "page" /2> 

尤其 需要 说 明 的 是 ,useBean 标记 中 scope 的 默认 值 是 page, KR page 之 外 ,scope 的 取 
值 还 有 request\session 与 application, 

1. scope 取 值 page 

该 bean 的 有 效 范围 是 当前 页 面 。 当 客户 请 求 bean 时 ,分 配 内 存 空间 给 它 , 当 客户 离开 
这 个 页 面 时 , 便 取消 分 配 的 bean, 并 收回 内 存 空 间 。JSP 引擎 分 配给 每 个 JSP 页 面 的 bean 
是 不 同 的 ,它们 占有 不 同 的 内 存 空间 。 

当 两 个 客户 访问 同一 个 JSP 页 面 时 ,一 个 用 户 对 自己 bean 的 属性 的 改变 ,不 会 影响 到 
男 一 个 客户 。 

2. scope 取 值 request 

该 bean 的 有 效 范围 是 request 期 间 。 客 户 在 网 站 访问 时 请 求 多 个 页 面 , 如 果 每 个 页 面 
都 含有 useBean 标记 ,那么 在 每 个 页 面 分 配 的 bean 也 不 相同 。JSP 引擎 对 请 求 做 出 响应 
后 ,bean 将 消失 。 


第 5 章 JavaBean 的 使 用 93 


当 两 个 客户 同时 请 求 一 个 JSP 页 面 时 ,一 个 用 户 对 自己 bean 属性 的 改变 ,不 会 影响 另 
外 一 个 客户 。 

3. scope 取 值 session 

该 bean 的 有 效 范围 是 客户 的 会 话 期 间 。 如 果 客 户 在 多 个 页 面 中 互相 连接 ,每 个 页 面 都 
含有 一 个 相同 的 useBean 标记 ,那么 这 个 客户 在 这 些 页 面 得 到 的 bean 是 相同 的 , 即 占有 相 
同 的 内 存 空 间 。 当 会 话 结束 时 ,bean 消失 ,释放 空间 。 

如 果 一 个 客户 在 某 个 页 面 更 改 了 bean 的 某 个 属性 ,那么 该 客户 的 其 他 页 面 bean 的 属 
性 也 发 生变 化 。 但 两 个 客户 同时 访问 一 个 JSP 页 面 时 ,一 个 客户 对 自己 bean 的 属性 的 改变 
不 会 影响 到 另 一 个 客户 。 

4. scope 取 值 application 

该 bean 的 有 效 范围 是 application $ [8] (Web 服务 器 启动 期 间 )。JSP 引擎 为 所 有 的 
JSP 页 面 分 配 一 个 共享 的 bean, 

当 几 个 客户 同时 访问 一 个 JSP 页 面 时 ,任何 一 个 客户 对 自己 bean 的 属性 的 改变 都 会 影 
响 到 其 他 客户 。 

当 含 有 useBean 动作 标记 的 JSP 页 面 被 JSP 引擎 (例如 Tomcat) 加 载 执行 时 ,JSP 引擎 
首先 根据 id 的 名 字 , 在 pageContext 内 置 对 象 中 查看 是 否 含有 名 字 id 和 作用 域 scope 的 对 
象 ; 如 果 该 对 象 存在 ,JSP 引擎 就 将 这 个 对 象 的 副本 (bean) 分 配给 JSP 页 面 使 用 ; 如 果 没 有 
找到 ,就 根据 class 指定 的 类 创建 一 个 名 字 是 id 的 bean, 并 添加 到 pageContext 对 象 中 , 同 
时 将 这 个 bean 分 配给 JSP 页 面 使 用 。useBean 动作 标记 执行 流程 如 图 5-3 所 示 。 


| useBean 标 记 执行 开始 | 


i 


在 usbBean 标 记 中 获得 id 、scope 的 值 


在 pageContext 中 
查找 指定 的 id、 
scope 值 的 bean 


1 


创建 指定 id 、scope 值 的 bean， 添 
加 到 pageContext 中 ， 并 给 客户 


给 客户 分 配 一 个 指定 id 、scope 值 的 bean 


usbBean 标 记 执行 结 


图 5-3 useBean 标记 执行 流程 图 
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5.2.2 技能 操作 


在 JSP 页 面 中 使 用 动作 标记 useBean, 具 体 任务 如 下 : 编写 一 个 JSP 页 面 eg5_1. jsp. TE 
JSP 页 面 中 使 用 useBean 标记 获得 一 个 bean. fi 9t 8| ££ bean 的 类 是 5. 1. 2 节 任 务 中 的 
Student 类 ,创建 bean 的 名 字 是 student. student 的 scope 取 值 为 page, 

代码 模板 eg5_1. jsp 如 下 : 


<% @ page language =" java" contentType 一 text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<%@ page import= "china. dalian. Student" % > 
<html> 
<head> 
<title>eg5_1.jsp</title> 
</head> 
<body bgcolor= "cyan"> 
二 %-- 通 过 useBean 标记 获得 一 个 bean, 负责 创建 bean 的 类 是 china. dalian. Student, id 是 student, 
scope 取 值 为 page --%> 
二 jsp:useBean id= "student" class= "china. dalian. Student" scope= "page" />> 
二 p> 该 生 的 学 号 是 : <% — student. getlIDO 47 
p» lt JE: <% =student. getName) ⁄> 
一 p 二 数学 成 绩 是 : <% — student. getMath() ⁄> 
一 p> 英 语 成 绩 是 : < %¿= student. getEnglish() 267 
<p> BJ E: <% = student. getSum() ⁄> 
</body> 
</html> http://localhost:8080/ch05/eg5_1.jsp| 


运行 上 面 的 eg5_1.jsp, 页 面 效果 如 图 5-4 所 示 。 

从 创建 bean 的 过 程 可 以 看 出 ,首次 创建 一 个 新 的 bean 
需要 用 相应 的 字 节 码 文 件 创建 对 象 , 当 别 的 JSP 页 面 再 需 
要 同样 的 bean 时 ,JSP 引擎 直接 将 pageContext 内 置 对 象 
里 已 经 存在 的 对 象 的 副本 分 配给 相应 的 JSP 页 面 ,提高 了 
代码 的 复 用 程度 。 如 果 程 序 员 修 改 了 字 节 码 文件 ,必须 重 
JA JSP 引擎 ,才能 使 用 新 的 字 节 码 文件 。 图 5-4 使 用 bean 的 JSP 页 面 


5.2.3 拓展 训练 


1. 下 面 ( ) 是 正确 使 用 JavaBean 的 方式 。 
A. —jsp:useBean id="address" class= "tom. AddressBean" scope="page"/> 


B. —jsp:useBean name= "address" class— "tom. AddressBean" scope— "page" />> 
C. —jsp:useBean bean= "address" class— "tom. AddressBean" scope— "page" /> 


D. —jsp:useBean beanName- "address" class=" AddressBean" scope— "page" /> 
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2. JavaBean 的 作用 域 可 以 是 ( ) page.session 和 application, 
A. request B. response C. out D. 以 上 都 不 对 
3. 在 J2EE,test.jsp 文件 中 有 如 下 一 行 代码 : 


<jsp:useBean class= "tom. jiafei. Test" id= "user" scope=" Ns 
要 使 user 对 象 一 直 存在 于 会 话 中 ,直至 终止 或 被 删除 为 止 ,下 划 线 中 应 填 人 ( Je 
A. page B. request C. session D. application 


4. XT JavaBean 正确 的 说 法 是 ( dis 
A. 类 中 声明 的 方法 的 访问 权限 必须 是 public 
B. 在 JSP 文 件 中 引用 bean, 其 实 就 是 用 一 jsp:useBean 二 语句 
C. bean 文件 放 在 任何 目录 下 都 可 以 被 引用 


D. 以 上 均 不 对 
5. 在 J2EE 中 ,test.jsp 文件 中 有 如 下 一 行 代码 : 
<jsp:useBean id=" user" scope= " " type=" com. UserBean" /> 
要 使 user 对 象 在 用 户 对 其 发 出 请 求 时 存在 ,下 划 线 中 应 填 入 ( Jis 
A. page B. request C. session D. application 


6. TE JSP 中 ,使 用 二 jsp:useBean 二 动作 可 以 将 JavaBean 引入 JSP 页 面 ,对 JavaBean 
的 访问 范围 不 能 是 ( m 


A. page B. request C. response D. application 
7. fE J2EE 中 test. jsp 文件 如 下 : 
<body> 


<jsp:useBean id = "buffer" scope= "page" class= "java. lang. StringBuffer" />> 
<% buffer. append(" ABC"); ⁄> 
buffer is <% = buffer% > 
</body> 
该 文件 试图 运行 时 ,将 发 生 ( )。 
A. 编译 期 间 发 生 错误 
B. 运行 期 间 抛 出 异常 [http://localhost:8080/ch05/testjsp — 
C. 运行 后 ,浏览 器 上 显示 : buffer is null 
D. 运行 后 ,浏览 器 上 显示 : buffer is ABC 
8. 编写 一 个 JSP Jt ifii computerRect. jsp, Æ JSP 页 面 中 使 
用 useBean 标记 获得 一 个 bean, 负 责 创建 bean 的 类 是 5. 1. 3 节 
中 的 Rect 类 ,创建 bean 的 名 字 为 rect, rect 的 scope 取 值 为 
request, JSP 页 面 的 运行 效果 如 图 5-5 所 示 。 要 求 将 如 下 代 5-5 ”使 用 bean 计算 和 矩形 
码 模板 补充 完整 ,并 调试 运行 。 的 周 长 和 面积 
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代码 模板 test. jsp 如 下 : 
<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
【代码 段 1]/ /使 用 page 指令 标记 的 import 属性 ,导入 Rect 类 
<head> 
<title>test. jsp</title> 
</head> 
<body bgcolor= "cyan"> 
REE 21//<%--38 it useBean 标记 获得 一 个 bean, 负 责 创建 bean 的 类 是 graph. picture. Rect, id 
是 rect, scope 取 值 为 page -- 967 
p AE BE dE: <% = rect. getLengthO X> 
过 p> 矩形 的 宽 是 : <% = rect. getWidth() 967 
p JE B JE] KAE: <% =rect. getGirth() 047 
p AE BS EU: <% —rect.getAreaO) 967 
</body> 
</html> 


5.3 获取 bean 属性 


JavaBean 的 实质 是 遵守 一 定 规范 的 类 所 创建 的 对 象 ,可 以 通过 以 下 两 种 方式 获取 bean 
的 属性 : 

1. Java 程序 片 

可 以 通过 调用 构造 方法 获得 一 个 bean, 然 后 调用 getXxx() 方 法 来 获取 bean 的 属性 。 

2. JSP 标记 

先 通 过 二 jsp:useBean 记 标记 获得 一 个 bean, BW iE — jsp: get Property > biu 2k C bean 
的 属性 值 (无 须 使 用 Java 程序 片 ) 。 


5.3.1 知识 要 点 
使 用 getProperty 动作 标记 可 以 获得 bean 的 属性 值 。 使 用 该 动作 标记 之 前 ,必须 事先 
使 用 useBean 动作 标记 获得 一 个 相应 的 bean, getProperty 动作 标记 语法 格式 如 下 : 
<jsp:getProperty name= "bean MAE" property= "bean 的 属性 " /> 
或 
<jsp:getProperty name= "bean 的 名 字 " property= "bean ff] JB fE" / >< /jsp:getProperty- 


其 中 ,name 取 值 是 bean 的 名 字 , 和 useBean 标记 中 的 id X4 property 取 值 是 bean 的 一 
个 属性 的 名 字 , 和 创建 该 bean 的 类 的 成 员 变 量 名 对 应 。 这 条 指令 相当 于 在 Java 表达 式 中 
使 用 bean 的 名 字 调 用 getXxx 方法 。 
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5.3.2 技能 操作 


使 用 getProperty 动作 标记 获得 bean 的 属性 ,具体 任务 如 下 : 

* 创建 bean 的 源 文件 NewStudent. java( 在 包 china. dalian 中 ) ,该 bean 的 作用 是 计算 
学 生 三 门 课程 (数学 、 英 语 、 政 治 ) 的 成 绩 总 分 和 平均 分 。 

。 编写 一 个 JSP 页面 useGetProperty. jsp. ÆI% JSP 页 面 中 使 用 useBean 标记 创建 一 
个 名 字 是 score 的 bean, 并 使 用 getProperty 动作 标记 获取 score 的 每 个 属性 的 值 。 
负责 创建 score 的 类 是 NewStudent 类 。 

1. 创建 bean 的 源 文件 NewStudent. java 

代码 模板 NewStudent. java 如 下 : 


package china. dalian; 
public class NewStudent{ 

int number; 

String name; 

double math, english, politics; 

double sum=0.0,average=0.0; 

public NewStudent() ( // 定 义学 生 类 的 构造 方法 
number=1; 
name= "superMan" ; 
math— 100.0; 
english = 100.0; 
politics=100. 0; 

) 

public int getNumber() í // 定 义 获 取 学 生 学 号 的 方法 
return number; 

) 

public void setNumber(int number) { // 定 义 修改 学 生 学 号 的 方法 
this. number = number; 

} 

public String getName() í 
return name; 

) 

public void setName( String name) | 
this. name — name; 

) 

public double getMath() ( 
return math; 

) 

public void setMath(double math) ( 
this. math = math; 

) 

public double getEnglish() ( 
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return english; 

) 

public void setEnglish(double english) ( 
this.english — english; 

) 

public double getPolitics() | 
return politics; 

) 

public void setPolitics(double politics) ( 
this. politics — politics; 

) 

public double getSum() ( 
sum- math english politics; 
return sum; 

) 

public double getAverage() ( 
average— sum/3; 
return average; 


) 


2. 编写 JSP 页 面 useGetProperty. jsp 
代码 模板 useGetProperty. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<%@ page import— "china. dalian. NewStudent" % > 
<html> 
<head> 
<title> useGetProperty.jsp< /title> 
</head> 
<body bgcolor= "cyan" > 
<h2> X Jé (i getProperty 标记 的 页 面 。 </h2> 
二 %-- 下 面 的 语句 功能 是 通过 useBean 标记 获得 一 个 bean, 负责 创建 bean 的 类 是 china. dalian 
. NewStudent, id 是 score, scope 取 值 为 page -- 2677 
—jsp:useBean id= "score" class= "china. dalian. NewStudent" scope— "page" /> 
<% score. setNumber(10) ; X> 
<% score. setName( " littlePerson") ; 677 
二 p 二 该 生 的 学 号 是 : —jsp:getProperty property "number" name= "score" /2> 
姓名 是 : 二 jsp:getProperty property= "name" name= "score" />2> 
p CENE: <jsp:getProperty property= "math" name= "score" /7 
英语 成 绩 是 : <jsp:getProperty property= "english" name= "score" />> 
政治 成 绩 是 : <jsp:getProperty property= "politics" name= "score" />> 
<p> ÄRME: <jsp:getProperty property= "sum" name= "score" /7* 


平均 分 是 : —jsp:getProperty property= "average" name= "score" />> 


</body> 
</html> 
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运行 JSP 页 面 useGetProperty. jsp ,效果 如 图 5-6 所 示 。 


http://localhost:8080/ch05/useGetPropertyjsp 


图 5-6 使 用 getProperty 标记 获得 bean 的 属性 值 


从 上 例 中 可 以 看 出 ,在 JSP 页 面 中 使 用 getProperty 动作 标记 获得 bean 的 属性 时 ,必须 
保证 bean 中 有 相应 的 getXxx 方法 , 即 创建 bean 的 类 中 定义 getXxx 方法 。 

从 useGetProperty. jsp 页 面 可 以 看 出 ,使 用 getProperty 动作 标记 获得 bean 的 属性 值 ， 
减少 了 Java 程序 片 的 使 用 。 

在 NewStudent. java 中 ,有 两 个 属性 (sum 和 average) 没 有 提供 set 方法 ,因为 二 者 依赖 
于 math .english 和 politics 属性 。 


5.3.3 拓展 训练 


1. 下 面 的 语句 中 ,与 二 jsp: getProperty name — "aBean" property — "jsp" /— 45 ff ff 
是 ( Js 
A. 6 —jspO > 
B. — out. printCaBean. getJspO) ; 47 
C. <% —aBean. setJspO % > 
D. < %aBean. setJspO ; 967 
2. Æ JSP 中 使 用 二 jsp:getProperty 二 标记 时 ,不 会 出 现 的 属性 是 ( Ja 
A. name B. property 
C. value D. D, E # A 2 h l 
3. 创建 bean 的 源 文 件 Ladder. java( 在 包 graph. picture 中 ) ,该 bean 的 作用 是 计算 梯 
形 的 面积 ; 编写 一 个 JSP 页 面 ladderProperty. jsp, 在 该 JSP 页面 中 使 用 useBean 标记 创建 
一 个 名 字 是 lad 的 bean ,并 使 用 getProperty 动作 标记 获得 lad 的 每 个 属性 的 值 。 负 责 创建 
lad 的 类 是 Ladder 类 。 
4. 创建 bean 的 源 文件 Circle. java( 在 包 graph. picture 中 ) ,该 bean 的 作用 是 计算 圆 形 
的 周 长 和 面积 ; 编写 一 个 JSP 页 面 circleProperty. jsp. Æ% JSP 页 面 中 使 用 useBean 标记 
创建 一 个 名 字 是 circle 的 bean ,并 使 用 getProperty 动作 标记 获得 circle 的 每 个 属性 的 值 。 
负责 创建 circle 的 类 是 Circle 35, 
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5.4 设置 bean 属性 


我 们 已 经 知道 ,可 以 通过 两 种 方式 获取 bean 的 属性 ,同样 修改 bean 的 属性 也 有 以 下 两 
种 方式 : 

1. Java 程序 片 

可 以 通过 调用 构造 方法 获得 一 个 bean, 然 后 调用 setXxx() 方 法 来 修改 bean 的 属性 。 

2. JSP 标记 

先 通过 二 jsp:useBean 二 标记 获得 一 个 bean, 再 通过 一 jsp:setProperty 二 标记 修改 bean 
属性 值 。 


5.4.1 知识 要 点 


使 用 setProperty 动作 标记 可 以 修改 bean 的 属性 值 。 使 用 该 动作 标记 之 前 ,必须 事先 
使 用 useBean 动作 标记 获得 一 个 相应 的 bean. ffi fH setProperty 动作 标记 进行 bean 属性 的 
设置 有 三 种 方式 : 

1. 用 表达 式 或 字符 串 设置 bean 的 属性 

(1) 用 表达 式 设 置 bean 的 属性 。 

<jsp:setProperty name= "bean 的 名 字 ”property 一 “bean 的 属性 " value= "< %⁄¿= expression% >" /> 

(2) 用 字符 串 设置 bean 的 属性 。 

—jsp:setProperty name= "bean 的 名 字 " property= "bean 的 属性 " value 一 字符 串 /二 

2. 通过 HTTP 表单 的 参数 的 值 设置 bean 的 属性 

—jsp:setProperty name= "bean 的 名 字 " property=" * " /> 

3. 任意 指定 请 求 参 数 设 置 bean 的 属性 

<jsp:setProperty name= "bean 的 名 字 ”property 一 "属性 名 ”param 一 "参数 名 "/ 二 


可 以 根据 自己 的 需要 ,任意 选择 传递 的 参数 ,请 求 参数 名 无 须 与 bean 属性 名 相同 。 
S.4.2 技能 操作 


使 用 setProperty 动作 标记 修改 bean 的 属性 ,具体 任务 如 下 : 

。 用 表达 式 或 字符 串 修改 bean 的 属性 。 

。 通 过 HTTP 表单 的 参数 的 值 设 置 bean 的 属性 。 

1. 用 表达 式 或 字符 串 修 改 bean 的 属性 

使 用 5. 3. 2 节 中 创建 的 bean 源 文件 NewStudent. java( 在 包 china. dalian 包 中 ), 前 面 
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已 经 介绍 过 ,该 bean 的 作用 是 计算 学 生 三 门 课程 (数学 英语、 政治 ) 的 成 绩 总 分 和 平均 分 。 
然后 创建 JSP 页 面 useSetProperty_1. jsp, ÆI% JSP 页 面 中 使 用 useBean 标记 创建 一 个 名 字 
是 score 的 bean, 其 有 效 范围 是 page, 并 使 用 动作 标记 修改 .获取 该 bean 的 属性 的 值 。 负 责 
创建 score 的 类 是 NewStudent 类 。 其 中 useSetProperty_1. jsp 的 页 面 效果 如 图 5-7 所 示 。 


http://localhost:8080/ch05/useSetProperty_1jsp 


图 5-7 ”使 用 字符 串 或 表达 式 的 值 修改 bean 的 属性 


代码 模板 NewStudent. java: 
参见 5. 3. 2 节 中 的 NewStudent. java, 
代码 模板 useSetProperty_1. jsp 如 下 : 


<% @ page language =" java" contentType 一 text/html; charset = GB] 

"GBK"%> 

<%@ page import= "china. dalian. NewStudent" % > 

<html> 

<head> 

<title> useSetProperty_1.jsp</title> 

</head> 

<body bgcolor= "cyan"> 

<h2> 3X Jé (ii Hl setProperty fiic ff] Uti. — /h27 

—jsp:useBean id= "score" class= "china. dalian. NewStudent" scope— "page" /> 
去 %%-- 使 用 setProperty 标记 设置 score ff) AR tE -> 


二 jsp:setProperty property= "number" name= "score" value= "12" />> 


pageEncoding = 


<jsp:setProperty property= "name" name= "score" value= "Johny" />> 


<jsp:setProperty property= "math" name= "score" value= "98.5" />2> 
<jsp:setProperty property= "english" name= "score" value= "78.0" />> 
<jsp:setProperty property= "politics" name= "score" value= "89.0" />> 
<p>E 5 Æ: <jsp:getProperty property— "number" name= "score" /2> 
姓名 是 : <jsp:getProperty property= "name" name=" score"/> 
<p>% RÆ: <jsp:getProperty property= "math" name= "score" /> 
英语 成 绩 是 : <jsp:getProperty property= "english" name= "score"/> 
政治 成 绩 是 : <jsp:getProperty property =" politics" name= "score"/> 
一 p 二 总 成 绩 是 : <jsp:getProperty property="sum" name= "score" />> 
平均 分 是 : —jsp:getProperty property= "average" name= "score" />> 
</body> 
</html> 


102 网 络 编程 与 动态 网 站 制作 


用 表达 式 修改 bean 属性 的 值 时 ,表达 式 值 的 类 型 必须 与 bean 的 属性 类 型 一 致 。 用 字 
符 串 修改 bean 属性 的 值 时 ,字符 串 会 自动 被 转化 为 bean 的 属性 的 类 型 ,不 能 转化 成 功 的 可 
能 会 抛 出 NumberFormatException 异常 。 


2. 通过 HTTP 表单 的 参数 的 值 设置 bean 的 属性 

编写 两 个 JSP 页 面 : inputMess. jsp 和 showMess. jsp. T£ inputMess. jsp 页 面 中 可 以 
输入 学 生 信息 (包括 学 号 、 姓 名、 数学 成 绩 、 英 语 成 绩 、 政 治 成 绩 ), 输 入 完毕 后 提交 给 
showMess. jsp 页 面 显示 信息 。 页 面 中 用 到 的 bean 仍然 是 使 用 5. 3.2 节 中 的 NewStudent 
类 来 创建 

代码 模板 NewStudent. java: 

参见 5. 3. 2 节 中 的 NewStudent. java, 

代码 模板 inputMess. jsp 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title> inputMess. jsp</title> 
</head> 
<body bgcolor=" cyan"> 
<form action= "showMess. jsp" method="post" > 
请 输入 学 生 学 号 : <input type="text" name= "number" /7 — br 
请 输入 学 生 姓 名 : <input type="text" name= "name" /7 — br 
请 输入 该 生 数学 课程 分 数 : <input type="text" name— " math"/7 — br 
请 输入 该 生 英 语 课程 分 数 : <input type="text" name= "english" /7 — br 
请 输入 该 生 政治 课程 分 数 : <input type="text" name= "politics" /7— br 
<input type="submit" value= "dà 2c" /77 
<input type— "reset" value— "8 # " />> 
</form> 
</body> 
</html> 


代码 模板 showMess. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"% > 

<%@ page import= "china. dalian. NewStudent" % > 

<% 

request. setCharacterEncoding( "GBK" ) ; 

n 

<html> 

<head> 

<title>showMess.jsp</title> 

</head> 
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odis 
—jsp:useBean id= "student" class= "china. dalian. NewStudent" scope— "page" />> 
二 %-- 通 过 HTTP 表单 的 参数 的 值 设置 bean f JR PE CRASAS JR VE Ë IU RO) > 
—jsp:setProperty property— " * " name= "student" /7 
<p> EM E: <jsp:getProperty property— "number" name= "student" />2> 
姓名 是 : <jsp:getProperty property= "name" name= "student" />> 
p CE WE: <jsp:getProperty property= "math" name= "student" />2> 
英语 成 绩 是 : 一 jsp:getProperty property= "english" name= "student" /> 
政治 成 绩 是 : 一 jsp:getProperty property= "politics" name= "student" /> 
<p> JE: 一 jsp:getProperty property= "sum" name= "student" />> 
平均 分 是 : —jsp:getProperty property= "average" name= "student" /7 
</body> 
</html> 


首先 运行 inputMess. jsp 页 面 ,效果 如 图 5-8 所 示 ; 当 用 户 输入 完 信息 单 击 * 提 交 ” 按 钮 
后 ,会 进入 showMess.jsp 页 面 , 如 图 5-9 所 示 。 


http;//localhost-:8080/ch05/inputMess jsp 


http://localhost:8080/ch05/showMess.jsp. 


该 生 的 学 号 是 ，5 姓名 是 ， 李 四 
数学 成 绩 是 ，87. 5 英语 成 绩 是 ，62. 0 政治 成 绩 是 ，95.5 
总 成 绩 是 ，245. 0 平均 分 是 ，81. 66666666666667 


图 5-8 信息 输入 页 面 图 5-9 信息 显示 页 面 


通过 HTTP 表单 的 参数 的 值 设 置 bean 的 属性 时 ,表单 的 参数 的 名 字 必 须 与 bean 属性 
的 名 字 相 同 , 服 务 器 会 根据 名 字 自 动 匹配 ,类 型 自动 转换 。 

由 于 客户 可 能 通过 表单 提交 汉字 字符 ,可 以 采用 request. setCharacterEncodingC" GBK ") ji 
免 出 现 中 文 乱码 。 采 用 该 方式 避免 中 文 乱码 时 ,表单 的 提交 方式 一 定 是 post 的 方式 。 


5.4.3 拓展 训练 


1. 在 JSP 中 调用 JavaBean 时 不 会 用 到 的 标记 是 ( js 
A. =javabean> B. <jsp:useBean> 
C. —jsp:setProperty-— D. —jsp:getProperty— 
2. 在 J2EE 中 ,以 下 是 有 关 jsp:setProperty 和 jsp:getProperty 标记 的 描述 ,其 中 正确 
的 是 ( Js 
A. <jsp:setProperty > fl —jsp:getProperty > bi iu $42 JI TE — jsp : useBean> fij JF 
始 标记 和 结束 标记 之 间 
B. 这 两 个 标记 的 name JB PERS (E 4^ 2L fI < jsp: useBean fiie HY id 属性 的 值 相对 应 
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C. 这 两 个 标记 的 name 属性 的 值 可 以 和 志 jsp:userBean 过 标记 的 id 属性 的 值 不 同 
D. 以 上 均 不 对 
3. 编写 两 个 JSP 页 面 : inputNumber. jsp 与 showResult. jsp. inputNumber. jsp 提供 
一 个 表单 ,用户 可 以 通过 表单 输入 两 个 数 和 四 则 运算 符号 提交 给 showResult. jsp。 用 户 提 
交 表 单 后 ,JSP 页 面 将 计算 任务 交 给 一 个 bean 去 完成 ,创建 bean 的 源 文 件 Computer. java 
(在 包 compute. add 中 ) 。 


5.5 JSP 与 JavaBean 结合 实例 


5.5.1 知识 要 点 


通过 前 面 的 学 习 已 经 知道 ,在 JSP 页 面 中 使 用 JavaBean 可 以 将 数据 的 处 理 代 码 从 页 面 
中 分 离 出 来 ,提高 了 代码 的 复 用 程度 ,方便 了 代码 的 维护 。 为 了 进一步 掌握 JavaBean 的 使 
用 方法 ,本 节 给 出 了 一 个 综合 性 的 实例 : 猜 数 字 游 戏 。 


5.5.2 技能 操作 


编写 两 个 JSP 页 面 : getNumber. jsp 与 guess. jsp. getNumber. jsp 页 面 能 够 产生 一 个 
1 一 100 之 间 的 随机 数 ,用 户 可 以 在 文本 框 中 给 出 自己 的 猜测 并 提交 给 guess. jsp 页 面 处 理 ; 
用 户 提 交 猜 测 后 guess. jsp 页 面 将 验证 猜测 的 任务 交 给 一 个 bean BI GuessNumber. java 去 
完成 ,该 bean 能 够 提示 用 户 “ 猜 大 了 ”或 是 “ 猜 小 了 "直至 “ 猜 对 了 ”为 止 ,同时 对 用 户 所 猜 次 
数 进行 统计 。 

代码 模板 GuessNumber. java 如 下 : 


package china. dalian; 
public class GuessNumber 


{ intanswer—0, // 待 猜测 的 整数 
guessNumber 一 0， // 用 户 的 猜测 
guessCount—0; // 用 户 猜测 的 次 数 


String result=null; 
boolean right= false; 
public void setAnswer(int n) 
{ answer—n; 
guessCount—0; 
) 
public int getAnswer() 
Í return answer; 
) 
public void setGuessNumbert(int n) 
Í guessNumber— n; 
guessCount+ + ; 
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if(guessNumber= =answer) 
{ result 一 "恭喜 , 猜 对 了 "; 
right— true; 
) 
else if(guessNumber--* answer) 
{ resut— EXT"; 
right— false; 
) 
else if(guessNumber-Canswer) 
( result "f^ T"; 
right— false; 
) 
) 
public int getGuessNumber( ) 
Í return guessNumber; 
) 
public int getGuessCount( ) 
Í return guessCount; 
) 
public String getResult() 
Í return result; 
) 
public boolean isRight() 
Í return right; 
) 
) 


代码 模板 getNumber. jsp 如 下 : 


<%@ page contentType= "text/html;charset=GB2312" % > 

<%@ page import= "china. dalian. GuessNumber" %> 

html <body> 

<% int n= (int) (Math. random() * 100) 4-1; % > 

—jsp:useBean id= "guess" class= "china. dalian. GuessNumber" scope— "session" /> 
—jsp:setProperty name= "guess" property— "answer" value="<%=n%>" /> 
二 p 二 随机 给 你 一 个 1 到 100 之 间 的 数 ,请 猜测 这 个 数 是 多 少 ? 

<% String str=response.encodeRedirectURL( "guess.jsp"); 

5n 

<form action="<% =str% >" method= post 

二 br 二 输入 你 的 猜测 : <input type= text name= "guessNumber"> 

<input type=submit value= "x" > 

</form></body> 

</html> 


代码 模板 guess. jsp 如 下 : 


<%@ page contentType— "text/html;charset— GB2312" ⁄—> 
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<%@ page import= "china. dalian. GuessNumber" % > 
<% String strGuess=response. encodeRedirectURL("guess.jsp"), 
strGetNumber- response. encodeRedirectURL("getNumber.jsp"); 
%> 
=html><body> 
<jsp:useBean id= "guess" class= "china. dalian. GuessNumber" scope= "session" /> 
<jsp:setProperty name= "guess" property= "guessNumber" param= "guessNumber" /> 
<br><jsp:getProperty name= "guess" property= "result" /二 ,这 是 第 
<jsp:getProperty name= "guess" property= "guessCount" /二 猜 . 
你 给 出 的 数 是 二 jsp:getProperty name= "guess" property= "guessNumber" /> 
<% if(guess.isRight()= = false) 
{ 
262» <form action="<% =strGuess% >" method= post 
再 输入 你 的 猜测 : <input type= text name= "guessNumber"> 
<input type=submit value= "di Zz "7 
</form> 
<% } 
> 
<br> 
<a href= "< % —strGetNumber / >"> [9] #J Ë 9t , 3R 3r JF fr cf 8k F UE XXL! 
</a> 
</body> 
</html> 


首先 运行 getNumber. jsp 页 面 ,如 图 5-10 Brzs ; 当 用 户 给 出 猜测 数字 "50” 并 单 击 “ 提 
交 ” 按 钮 后 ,出 现 如 图 5-11 所 示 的 页 面 效果 ; 直至 最 后 将 数字 猜 对 ,会 出 现 如 图 5-12 所 示 的 
页 面 效果 。 


http://localhost:8080/ch05/getNumber jsp 


随机 给 你 一 个 1 到 100 之 间 的 数 ， 请 猜测 这 个 数 是 多 少 ? 


输入 你 的 猜测 : 50 


5-10 输入 数字 页 面 


http://localhost:8080/ch05/guessjsp 


猜 大 了 , 这 是 第 1 猜 . 你 给 出 的 数 是 50 
再 输入 你 的 猜测 : [E53] 


http;//localhost:8080/ch05/guess.jsp 


PE, WAT ESE 6H. HALDRE 39 
加 到 首页 ， 重 新 开始 玩 靖 未 字 游 戏 ! 


回 到 首页 ， 重 新 开始 玩 猜 数字 游戏 ! 


图 5-11 猜测 过 程 页 面 图 5-12 猜 对 数字 页 面 
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5.5.3 拓展 训练 


运用 本 章 所 学 的 JavaBean 相关 内 容 ,修改 4. 2. 3 节 中 第 7 题 的 简易 计算 器 。 具 体 要 求 
如 下 : 编写 JSP 页 面 caculator. jsp ,在 该 页 面 中 用 户 可 以 输入 两 个 运算 数 ,可 以 选择 加 、 减 、 
乘 、 除 运算 符 进行 运算 ; 页 面 将 计算 的 任务 提交 给 一 个 bean 即 Calculator. java 去 完成 。 页 
面 运行 效果 如 图 5-13 所 示 。 


http://localhost;8080/ch05/calculatorjsp 


简单 的 计算 器 
请 输入 第 一 个 运算 数 45 
请 选择 所 要 做 的 运算 [加 
请 输入 第 二 个 运算 数 2 


计算 结果 是 :45 * 2 =90 


513 简易 计算 器 


5.6 J 结 


。 JavaBean 的 实质 是 一 种 特殊 的 Java 类 ,是 一 个 可 重复 使 用 的 软件 组 件 ,是 遵循 一 定 
标准 .用 Java 语言 编写 的 一 个 类 ,该 类 的 一 个 实例 称 为 一 个 JavaBean ,简称 bean, 


* JSP 和 JavaBean 技术 的 结合 不 仅 可 以 实现 数据 的 表示 和 处 理 分 离 , 而 且 提高 了 代码 
的 可 重用 、 可 维护 性 。 


。 JavaBean 的 生命 期 限 分 为 page、request、session 和 application, 


LIES 
JSP 对 数据 库 的 访问 


数据 库 在 现在 的 Web 应 用 中 扮演 着 越 来 越 重 要 的 作用 。 如 果 没 有 数据 库 ,很 多 重要 的 
应 用 , 像 电子 商 务 、 搜 索引 擎 等 都 不 可 能 实现 。 本 章 主要 介绍 在 JSP 中 如 何 访问 关系 数据 
库 , 如 Oracle、SQL Server, MySQL 和 Microsoft Access 等 数据 库 。 

本 章 将 新 建 一 个 Web 工程 ch06, 本 章 例 子 中 涉及 的 Java 源 文件 保存 在 ch06 的 src 中 ， 
涉及 的 JSP 页 面 保存 在 ch06 的 WebContent 中 。 


6.1 使 用 JDBC-ODBC 桥接 数据 库 


JSP 页 面 中 访问 数据 库 , 首 先 要 与 数据 库 进行 连接 ,通过 连接 向 数据 库 发 送 指令 ,并 获 
得 返回 的 结果 。JDBC 连接 数据 库 通常 有 两 种 常用 方式 : 建立 JDBC-ODBC 桥接 器 和 加 载 
纯 Java 驱动 程序 ,本 节 主 要 介绍 前 一 种 ,6. 2 节 将 介绍 第 二 种 。 


6.1.1 知识 要 点 


JDBC(Java DataBase Connectivity) 是 用 于 运行 SQL 的 解决 方案 ,是 Java 运行 平台 核 
心 类 库 中 的 一 部 分 , 它 由 一 组 标准 接口 与 类 组 成 。 我 们 经 常 使 用 JDBC 完成 3 件 事 : 与 指 
定 的 数据 库 建 立 连接 ; 向 已 连接 的 数据 库 发 送 SQL 命令 ; 处 理 SQL 命令 返回 的 结果 。 

ODBC(Open DataBase Connectivity) 是 由 Microsoft 主导 的 数据 库 连 接 标 准 , 提 供 了 通 
用 的 数据 库 访问 平台 。 但 是 ,使 用 ODBC 连接 数据 库 的 应 用 程序 移植 性 较 差 , 因 为 应 用 程 
序 所 在 的 计算 机 必须 提供 ODBC. 

使 用 JDBC-ODBC 桥接 器 连接 数据 库 的 机 制 是 : 将 连接 wema) 
数据 库 的 相关 信息 提供 给 JDBC-ODBC 驱动 程序 ,然后 转换 成 
JDBC 接口 ,供应 用 程序 使 用 ,而 和 数据 库 的 连接 是 由 ODBC JDBC 应 用 程序 接口 
完成 。 使 用 JDBC-ODBC 桥接 器 连接 数据 库 的 示意 图 如 图 6-1 JDBC-ODBC 桥 接 器 
所 示 。 

使 用 JDBC-ODBC 桥接 器 连接 数据 库 有 3 个 步骤 : 

(1) 建立 JDBC-ODBC 桥接 器 。 

(2) 创建 ODBC 数据 源 。 

(3) fll ODBC 数据 源 指定 的 数据 库 建立 连接 。 图 6-1 JDBC-ODBC 桥接 器 


ODBC 驱 动 程序 


IODBC 数 据 源 (X x X 数据 库 
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6.1.2 技能 操作 


运用 JDBC-ODBC 桥接 器 连接 数据 库 , 具 体 任务 如 下 : 

。 创建 待 连接 的 Microsoft Access 数据 库 。 

* 建立 JDBC-ODBC 桥接 器 。 

。 创建 ODBC 数据 源 。 

* 和 ODBC 数据 源 指定 的 数据 库 建立 连接 。 

。 在 JSP 页 面 中 使 用 JDBC-ODBC 桥接 器 连接 数据 库 。 

1. 创建 待 连接 的 Microsoft Access 数据 库 

使 用 Microsoft Access 2007 设计 一 个 教师 数据 库 teacher DB 并 保存 到 “D:\” 盘 根 日 
录 ,该 库 中 有 一 张 教师 信息 表 teacherInfo, 表 的 字段 如 图 6-2 所 示 。 


p 字段 名 称 ， BRAM 

V teacherID 文本 教师 编号 
jteacherNane 文本 教师 姓名 
teacherLevel 文本 教师 职称 

teaherTine 文本 教师 工作 时 间 
teacherSalary 货币 教师 工资 


图 6-2 teacherInfo 表 的 说 明 


表 中 的 数据 如 图 6-3 所 示 。 


赵 晓 晓 助教 2015/3/1 X2, 500. 00 


1 
2 钱 多 多 讲师 2012/9/1 13, 000. 00 
3 孙 苗 苗 副教授 2002/3/5 x3, 800. 00 
+ 李 大 海 副教授 2000/3/1 14, 000. 00 
5 周正 教授 1994/9/10. X5, 500, 00 


图 6-3 teacherInfo 表 中 的 数据 


2. 创建 ODBC 数据 源 

创建 ODBC 数据 源 时 ,必须 保证 计算 机 有 ODBC 系统 , Windows 操作 系统 一 般 都 带 有 
ODBC 系统 。 

1) 打开 ODBC 数据 源 管理 器 

在 Windows 7 系统 中 ,在 “控制 面板 ”中 选择 “系统 和 安全 ”一 “管理 工具 ”选项 (有 些 
Windows XP 系统 , 需 在 “控制 面板 ”中 选择 “性 能 和 维护 ”>“ 管 理工 具 ” 选 项 ) ,找到 “数据 源 
(ODBC) ”图 标 双 击 打开 ,出 现 如 图 6-4 所 示 的 界面 。 

2) 为 数据 源 选 择 驱 动 程序 

在 图 6-4 所 示 的 界面 选择 “用 户 DSN ?选项 卡 , 单 击 * 添 加 ?按钮 ,出 现 为 新 增 的 数据 源 选 
择 驱 动 程序 界面 ,如 图 6-5 所 示 。 因 为 我 们 要 连接 Microsoft Access 2007 数据 库 , 所 以 选择 
“Microsoft Access Driver( * . mdb, * .accdb)”, 单 击 “ 完 成 ”按钮 。 
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dBASE Files Microsoft Access dBASE Driver (4. wD | 
Excel Files Microsoft Excel Driver (* xls, * x] = 


MS Access Database Microsoft Access Driver (*.mdb, *. 


Enum lad) 


FERAE MEERA aa A 


[确定 ][ má 应 用 (A) 帮助 


图 6-4 打开 ODBC 数据 源 管理 器 


| 选择 您 想 为 其 安装 数据 源 的 9 动 程序 © | 
名 称 š 
Driver do Microsoft dBase (# dbf) 
Driver do Microsoft Excel (+. xls) B il 


Driver do Microsoft Paradox (db ) 

Driver para o Microsoft Visual FoxPro 
Microsoft Access dBASE Driver (*.dbf, *.n& 
Microsoft Access Driver (* mdb) 
Microsoft Acces iver (*.mdb, *.accdb) 
Microsoft Access Paradox Driver (*. db) 
< 


CCom (moi. 


65 ”为 新 增 的 数据 源 选择 驱动 程序 


3) 为 数据 源 起 名 并 找到 对 应 的 数据 库 

在 如 图 6-5 所 示 的 界面 单 击 “ 完 成 ”按钮 出 现 设置 数据 源 具 体 信息 的 对 话 框 ,如 图 6-6 
所 示 。 在 “数据 源 名 ”文本 框 中 为 数据 源 起 个 名 字 : teacher。 在 图 6-6 中 单 击 “ 选 择 ” 按 钮 为 
teacher 数据 源 选择 数据 库 “D:\teacherDB. accdb” 之 后 确定 ,如 图 6-7 所 示 。 

4) 设置 登录 名 和 密码 

回 到 如 图 6-6 所 示 的 界面 , 单 击 “ 高 级 "按钮 ,出 现 设 置 登录 名 与 密码 界面 ,如 图 6-8 所 
示 。 这 里 的 用 户 名 是 tt, 密 码 是 123。 在 如 图 6-8 所 示 的 界面 单 击 * 确 定 ” 按 钮 后 ,再 单 击 如 
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6-7 选择 数据 库 


图 6-6 所 示 的 界面 中 的 “确定 ”按钮 就 创建 了 一 个 新 的 数据 源 : teacher, 看 到 如 图 6-9 所 示 的 
界面 就 表示 数据 已 创建 完成 。 


iver 
ExtendedAnsiSQL 


0 
MS Access; 


Inpli ci tConmi tSyne 
Masna £ €. 2048 


Ew DefaultDir D 


68 设置 登录 名 和 密码 


112 网 络 编程 与 动态 网 站 制作 


= ODBC =: "- 
用 户 ISI [系统 pss | osu | 驱动 程序 | gy [ika | 关于 _ 
用 户 数据 源 QD: 

E wi 
dBASE Files Microsoft Access dBASE Driver (+. d] 
Excel Files Microsoft Excel Driver (*.xls, * x| 
MS Access Database Microsoft Access Driver (*.mdb, +. 
teacher Microsoft Access Driver (*.mdb, *. 
 — n— . 

| 

0DBC 用 户 数 了 如 何 与 信息 。 用 


AE (98 J Emo [Wm J 


6-9 完成 数据 源 的 创建 


3. 建立 JDBC-ODBC 桥接 器 

JDBC 通过 java. lang. Class 类 的 静态 方法 forName 加 载 sun. jdbc. odbc. JdbcOdbcDriver 类 
建立 JDBC-ODBC 桥接 器 。 建 立 桥接 器 时 可 能 发 生 ClassNotFoundException 异常 ,必须 捕 
获 该 异常 ,建立 桥接 器 的 具体 代码 如 下 ; 


try{ 

Class. forName( "sun. jdbc. odbc. JdbcOdbcDriver") ; 
) catch( ClassNotFoundException e) ( 

e. printStackTrace() ; 


) 


4. 与 ODBC 数据 源 指 定 的 数据 库 建立 连接 
首先 ,使 用 java. sal 包 中 的 Connection 类 声明 一 个 连接 对 象 con ,然后 再 使 用 java. sql 
包 中 的 DriverManager 类 调用 静态 方法 getConnection 创建 连接 对 象 con: 


Connection con— DriverManager. getConnection("jdbc:odbc: 数 据 源 名 字 ", "登录 名 ", "密码 "); 
如 果 没 有 给 数据 源 设置 登录 名 和 密码 ,那么 连接 形式 是 : 
Connection con= DriverManager. getConnection("jdbc:odbc: 数 据 源 名 字 ","",""); 


建立 连接 时 应 捕获 SQLException 异常 ,如 ,和 数据 源 myGod 指定 的 数据 库 goods. accdb i 
立 连接 代码 如 下 : 
try{ 


Connection con= DriverManager. getConnection( "jdbc:odbc:myGod", "firstDB" , "firstDB") ; 
) catch( SQLException e) { 
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e. printStackTrace() ; 


) 
5. 在 JSP 页 面 中 使 用 JDBC-ODBC 桥接 器 连接 数据 库 


编写 一 个 JSP 页 面 eg6_1. jsp, 该 页 面 中 的 Java 程序 片 代码 使 用 JDBC-ODBC 桥接 器 
连接 到 数据 源 teacher. rif teacherInfo 表 中 的 全 部 记录 。 页 面 运 行 效果 如 图 6-10 所 示 。 


http;//localhost:8080/ch06/eg6 1jsp 


教师 编号 教师 姓名 BURE 入 职 时 间 | 教师 工资 
1 prm BDA 2015-03-01 00:00:00 2500. 0000 
钱 多 多 ”| 讲师 2012-09-01 00:00:00|3000. 0000 
Hh [mE [2002-03-05 00:00:00 [3800. 0000 
EAS [RRE [2000-03-01 00:00:00 [4000. 0000 
周正 RE 1994-09-10 00:00:00 5500. 0000 


A| =] €] t3! 


6-10 使 用 JDBC-ODBC 桥接 器 连接 数据 库 
代码 模板 eg6_1. jsp H F: 


<%⁄@ page language— "java" contentType= "text/html; charset=GBK" 
pageEncoding= "GBK" /4 > 
<%@ page import="java. sql. * "%> 
<html> 
<head> 
<title>eg6_1.jsp</title> 
</head> 
<body bgcolor= "lightyellow" > 
<% 
Connection con = null; 
Statement st = null; 
ResultSet rs = null; 
try | 
Class. forName(" sun. jdbc. odbc. JdbcOdbcDriver") ; 
} catch (ClassNotFoundException e) { 
e. printStackTrace() ; 
) 
try { 
con = DriverManager. getConnection( "jdbc:odbc:teacher" , "tt", "123"); 
st— con. createStatement() ; 
rs— st. executeQuery("select * from teacherInfo") ; 
out. print("< table border- 17"); 
out. print(" tr"); 
out. print(" th 3 lii 4j S — / th") ; 
out. print(" th E Mi f & < /th") ; 
out. print(" th JE Wf BUE / th") ; 
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out. print( "<th> A ERIS [8]  / th") ; 
out. print(" <th> Zi T yE-— /th ") ; 
out. print("</tr>"); 
while(rs. next) | 
out. print(" tr"); 
out. print( "<td>" 4-rs. getString( 1) +"</td>"); 
out. print( "<td>" + rs. getString( 2) +"</td>"); 
out. print( "<td>" 4 rs. getString( 3) +"</td>"); 
out. print( "<td>" rs. getString( 4) +"</td>"); 
out. print( "<td>" 4- rs. getString( 5) T " /td77") ; 
out. print("</tr>"); 
) 
out. print("  /table") ; 
) catch (SQLException e) í 
e. printStackTrace() ; 
) finally( 
try( 
if(rs! =null) í 
rs.close() ; 
) 
ifCst! — nul ( 
st.close() ; 
) 
if(con! — null) ( 
con. close() ; 
) 
)catch (SQLException e) | 
e. printStackTrace() ; 


) 

5 
</body> 
</html> 


6.1.3 拓展 训练 


1. 当 在 JSP 文 件 中 要 编写 代码 连接 数据 库 时 ,应 在 JSP 文件 中 加 入 以 下 哪个 请 
句 ?( ) 
—jsp:include file= "java. util. *"/> 
<% (9 page import= "java. sql.*" %> 
—jsp:include page= "java. lang. *"/> 
<% (page import = "java. util. *" Yi > 


口 P F > 
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2. Java 程序 连接 数据 库 常 用 的 两 种 方式 : 建立 JDBC-ODBC 和 加 载 纯 ( ) 驱动 

程序 。 
A. Oracle B. Java C. Java 数据 库 D. 以 上 都 不 对 

3. JDBC 连接 数据 库 常 用 的 方式 有 哪些 ? 

4. 参考 6. 1. 2 节 中 的 主要 内 容 , 创 建 数据 源 mySource, 该 数据 源 指定 的 数据 库 是 
teacherDB. accdb 。 

编写 一 个 JSP 页面 practice6_1. jsp, 该 页 面 中 的 Java 程序 片 代 码 使 用 JDBC-ODBC ffr 
接 器 连接 到 数据 源 mySource, 查 询 teacherInfo 表 中 工资 高 于 3000 元 的 教师 信息 ( 即 
teacherSalary 字段 值 大 于 3000 的 全 部 记录 )。 页 面 运 行 效果 如 图 6-11 所 示 。 


http://localhost:8080/chO6/practiceó 1jsp 


教师 编号 教师 姓名 BIRER 入 职 时 间 教师 工资 
3 HE BK [2002-03-05 00:00:00 5800. 0000 
4 EAS Bik [2000-03-01 00:00:00 4000. 0000 
5 周正 BRE [1994-09-10 00:00:00 5500. 0000 


6-11 practice 1.jsp 页 面 运行 效果 


6.2 使 用 纯 Java 数据 库 驱 动 程序 连接 数据 库 


6.2.1 知识 要 点 
应 用 程序 除了 使 用 JDBC-ODBC 桥接 器 连接 数据 库 外 ,还 通常 使 用 JDBC API 调用 本 


地 的 纯 Java 数据 库 驱 动 程序 和 相应 的 数据 库 建 立 连 接 , 如 - 

图 6-12 所 示 。 应 用 程序 (J2EE 应 用 ) 
使 用 纯 Java 数据 库 驱 动 程序 连接 数据 库 , 需 要 经 过 两 个 

步骤 ， JDBC API 
* 注册 纯 Java 数据 库 驱 动 程序 。 
”和 和 指定 的 数据 库 建立 连接 。 iava REE 
下 面 以 Oraclelog 为 例 , 讲 解 如 何 使 用 纯 Java 数据 库 驱 

动 程序 连接 数据 库 的 方法 。 
1. 注册 纯 Java 数据 库 驱 动 程序 数据 库 


每 种 数据 库 都 配 有 自己 的 纯 Java 数据 库 驱 动 程序 。 
OraclelOg 的 纯 Java 驱动 程序 一 般 位 于 数据 库 安装 目录 “\oracle\ 
product\10. 2.0\db_1\jdbc\lib” 下 ,名 为 classes12. jar, 


为 了 连接 Oraclelog 数据 库 , 可 以 将 classes12. jar 文件 复制 到 Web 应 用 程序 的 /WEB- 


图 6-12 使 用 纯 Java 数据 库 
驱动 程序 


116 网 络 编程 与 动态 网 站 制作 


INF/lib 文件 夹 中 。 然 后 ,通过 java. lang. Class 类 的 forName() ,动态 注册 Oraclel0g 的 纯 
Java 驱动 程序 ,代码 如 下 : 


try { 

Class. forName( "oracle. jdbc. driver. OracleDriver") ; 
) catch (ClassNotFoundException e) { 

e. printStackTrace() ; 


) 
2. 和 指定 的 数据 库 建立 连接 
和 Oracle 数据 库 建立 连接 的 代码 如 下 : 
try { 
Connection con= DriverManager. getConnection( "jdbc:oracle:thin:@ 主 机 :端口 号 :数据 库 名 "， 
"用 户 名 ", "密码 "); 
} catch (SQLException e) { 
e. printStackTrace() ; 
) 
其 中 ,主机 是 安装 Oracle 服务 器 的 TP 地 址 ,如 果 是 本 机 则 为 localhost; Oracle 默认 端口 号 
Jy 1521; Oracle 默认 数据 库 名 为 orcl; 用 户 名 和 密码 是 访问 Oracle 服务 器 的 用 户 权限 。 
本 章 中 后 续 的 例子 ( 除 6. 5 节 与 6.7 节 ) 均 采用 纯 Java 数据 库 驱 动 程序 连接 
Oraclel0g, 


6.2.2 技能 操作 


使 用 纯 Java 数据 库 驱 动 程序 连接 数据 库 , 具 体 任务 如 下 : 
* 利用 Oracle 建立 商品 数据 库 goods, 在 该 数据 库 中 建立 商品 信息 表 goosInfo， 
goosInfo 表 的 字段 如 表 6-1 所 示 。 


表 6-1 商品 信息 表 
字段 名 类 型 长 度 限制 备注 
goodsId number (4) 非 空 ,主键 商品 编号 
goodsName varchar (50) 非 空 商品 名 称 
goodsPrice number (7,2) 非 空 商品 价格 
goodsType varchar (10) 非 空 商品 类 型 


° 编写 一 个 JSP 页 面 eg6_2.jsp, 该 页 面 中 的 Java 程序 片 代码 使 用 纯 Java 驱动 程序 连 
接 Oracle 数据 库 ,查询 goodsInfo 表 中 的 全 部 记录 。 


创建 goodsInfo 表 的 SQL 语句 如 下 : 


= 


create table goodsinfo ( 
goodsId number(4) not null, 
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goodsName varchar(50) not null, 

goodsPrice number(7,2) not null, 

goodsType varchar(10) not null, 

constraint pk goodsinfo primary key (goodsld) 
); 
insert into goodsinfo values(1, 7 È ',12,' H Hm’); 
insert into goodsinfo values(2, "冰箱 ',2500, 8 2$) ; 
insert into goodsinfo values(3, "蛋糕 ',28, rà); 
insert into goodsinfo values(4, ' 苹 果 ', 48, KAN; 
insert into goodsinfo values(5, "LAK, 1800, "RAN; 
insert into goodsinfo values(6, ' 书 包 ',78, KAN; 
commit; 


页 面 运行 效果 如 图 6-13 所 示 。 


http://localhost:8080/ch06/eg6_2jsp 


图 6-13 ”使 用 纯 Java 驱动 程序 连接 Oracle 数据 库 
代码 模板 eg6_2. jsp MIT: 


— (3 page language— "java" contentType= "text/html; charset= GBK" 
pageEncoding— "GBK" % > 
<%@ page import="java. sql. * "%> 
<html> 
<head> 
<title>eg6_2.jsp</title> 
</head> 
<body bgcolor= "LightPink" > 
<% 
Connection con = null; 
Statement st = null; 
ResultSet rs = null; 
try { 
Class. forName( "oracle. jdbc. driver. OracleDriver") ;// 注 册 Oracle 的 纯 Java 驱动 程序 
) catch (ClassNotFoundException e) ( 
e. printStackTrace() ; 


) 
try ( 
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con 一 DriverManager. getConnection( "jdbc : oracle: thin: @localhost:1521:orcl", 
"system" , " system") ; 
st— con. createStatement() ; 
rs= st. executeQuery("select * from goodsInfo") ; 
out. print(" — table border— 177"); 
out. print(" — tr"); 
out. print( "<th> 8 à 4i 5 — / th"); 
out. print(" <th> Hi ñ R< / th") ; 
out. print( "<th> Hj ib ffr Fi — / tho ") ; 
out. print( "< th> Ñ à AH — / th ") ; 
out. print("  /tr") ; 
while(rs. next 1 
out. print(" tr"); 
out. print( "<td>" 4-rs. getString( 1) + "< /td 7"); 
out. print( "<td>" + rs. getString( 2) +"</td>"); 
out. print( "<td>" + rs. getString( 3) +"</td>"); 
out. print( "<td>" + rs. getString( 4) +"</td>"); 
out. print("</tr>"); 
) 
out. print("  /table") ; 
) catch (SQLException e) í 
e. printStackTrace() ; 
) finallyt 
try{ 
if(rs! = null) ( 
rs. close) ; 
) 
if(st! =null) ( 
st. close) ; 
) 
if(con! = null) ( 
con. close() ; 
) 
}catch (SQLException e) | 
e. printStackTrace() ; 


) 

> 
</body> 
</html> 


应 用 程序 连接 Oracle 数据 库 时 ,必须 事先 启动 Oracle 服务 器 的 OracleServiceORCL 和 
OracleOraDb10g_home1 TNSListener 两 个 服务 .否则 会 抛 出 连接 异常 。 
从 任务 中 可 以 看 出 编写 程序 访问 数据 库 需 要 有 以 下 几 个 步骤 : 
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(1) 导入 java. sql 包 。 

所 有 与 数据 库 有 关 的 对 象 和 方法 都 在 java. sal 包 中 , 包 java. sal 包含 了 用 Java 操作 关 
系数 据 库 的 类 和 接口 。 因 此 在 使 用 JDBC 的 程序 中 必须 要 加 入 import java. sql. * 。 

(2) 加 载 驱动 程序 。 

在 该 任务 中 使 用 了 Class 类 (java. lang 包 ) 中 的 方法 forName, 来 装 入 该 驱动 程序 的 类 
定义 oracle. jdbc. driver. OracleDriver, 从 而 创建 了 该 驱动 程序 的 一 个 实例 。 

(3) 连接 数据 库 。 

完成 上 述 操 作 后 ,就 可 以 连接 一 个 特定 的 数据 库 了 。 这 需要 创建 Connection 类 的 一 个 
实例 ,并 使 用 DriverManager 的 方法 getConnection 来 尝试 建立 用 url 指定 的 数据 库 的 连 
接 。 代 码 如 下 : 


con= DriverManager. getConnection( "jdbe: oracle: thin: (localhost: 1521 : orcl" , "system", "system") ; 


(4) 访问 数据 库 。 

访问 数据 库 时 ,需要 先 用 Connection 类 的 createStatement 方法 从 指定 的 数据 库 连 接 得 
到 一 个 Statement 的 实例 ,然后 用 这 个 实例 的 executeQuery 方法 来 执行 一 条 SQL 语句 。 代 
码 如 下 : 


st= con. createStatement( ) ; 


rs— st. executeQuery("select * from goodsInfo") ; 


(5) 处 理 返回 的 结果 集 。 
ResultSet 对 象 是 JDBC 中 比较 重要 的 一 个 对 象 ,几乎 所 有 的 查询 操作 都 将 数据 作为 
ResultSet 对 象 返 回 。 处 理 结果 集 ResultSet 对 象 的 代码 如 下 : 


while(rs.next()) Í 
out. print(" — tr"); 
out. print( "<td>" 4- rs. getString( 1) - "< /td>"); 
out. print( "<td>" 4 rs. getString(2) +"</td>"); 
out. print("< td>" 4- rs. getString(3) + "<< /td ^"); 
out. print( "<td>" 4- rs. getString(4) + " —/ td"); 
out. print("  /tr»"); 

) 


(6) 关闭 数据 库 连接 ,释放 资源 。 

对 数据 库 的 操作 完成 之 后 ,要 及 时 关闭 ResultSet 对 象 .Statement 对 象 和 数据 库 连 接 
对 象 Connection, 从 而 释放 占用 的 资源 ,这 就 要 用 到 close 方法 。 代 码 如 下 : 

rs. close) ; 


st. close); 
con. close() ; 


关闭 的 顺序 从 先 到 后 依次 为 ResultSet X A , Statement 对 象 和 Connection 对 象 。 
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6.2.3 拓展 训练 


使 用 纯 Java 数据 库 驱 动 程序 访问 数据 库 时 ,都 有 哪些 步骤 ? 

编写 一 个 JSP 页 面 practice6_2. jsp, 该 页 面 中 的 Java 程序 片 代码 使 用 纯 Java 驱动 程序 
连接 Oracle 数据 库 ,查询 goodsInfo 表 中 goodsPrice 字段 值 大 于 10 并 小 于 50 的 全 部 记录 。 
页 面 运 行 效果 如 图 6-14 所 示 。 


http://localhost:8080/ch06/practice6_2jsp 


6-14 practice6_2. jsp 页 面 运行 效果 


6.3 使 用 Statement 和 ResultSet 操作 数据 


6.3.1 知识 要 点 


与 数据 库 建立 连接 之 后 , 若 接 下 来 要 执行 SQL 语句 ,需要 进行 以 下 几 个 步骤 。 

1. 创建 Statement 对 象 

Statement 对 象 代表 一 条 发 送 到 数据 库 执行 的 SQL 语句。 由 已 创建 的 Connection 对 
象 con 调用 createStatement() 方 法 来 创建 Statement 对 象 ,代码 如 下 : 


Statement smt— con. createStatement() ; 


2. 执行 SQL 语句 

创建 Statement 对 象 之 后 ,可 以 使 用 Statement 对 象 调 用 executeUpdate(String sql), 
executeQuery( String sql) 等 方法 来 执行 SQL 语句 。 

executeUpdate(String sql) 方 法 主要 用 于 执行 INSERT .UPDATE 或 DELETE 语句 以 
及 SQL DDL 语句 ,例如 CREATE TABLE 和 DROP TABLE。 该 方法 返回 一 个 整数 (代表 
被 更 新 的 行 数 ) ,对 于 CREATE TABLE 和 DROP TABLE 等 不 操作 行 的 指令 ,返回 零 。 

executeQuery(String sql) 方 法 则 是 用 于 执行 SELECT 等 查询 数据 库 的 SQL 语句 ,该 
方法 返回 ResultSet 对 象 ,代表 查询 的 结果 。 

3. 处 理 返 回 的 ResultSet 对 象 

ResultSet 对 象 是 executeQuery(String sqlD) 方 法 的 返回 值 ,被 称 为 结果 集 , 它 代表 符合 
SQL 语句 条 件 的 所 有 行 。ResultSet 对 象 调用 next() 方 法 移动 到 下 一 个 数据 行 (顺序 查 
询 ) , 当 数据 行 存在 时 ,next() 方 法 返回 true, 和 否则 返回 false。 获 得 一 行 数据 后 ,ResultSet 对 
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象 可 以 使 用 getXxx 方法 获得 字段 值 ,getXxx 方法 都 提供 依 字段 名 称 取得 数据 ,或 是 依 字段 
顺序 取得 数据 的 方法 。 


6.3.2 技能 操作 


使 用 Statement 与 ResultSet 对 象 对 数据 库 进行 增删 改 查 ,具体 任务 如 下 : 编写 两 个 
JSP 页 面 一 一 addGoods. jsp 和 showAllGoods. jsp。 用 户 可 以 在 addGoods. jsp 页 面 中 输入 
信息 后 , 单 击 “ 添 加 ”按钮 把 信息 添加 到 goodsInfo 表 中 。 然 后 ,在 showAllGoods. jsp 页 面 
中 显示 所 有 商品 信息 。 在 该 任务 中 需要 编写 一 个 bean(GoodsBean. java) 用 来 实现 添加 和 
查询 记录 。 页 面 运行 效果 如 图 6-15 和 图 6-16 所 示 。 


http://localhost:8080/ch06/addGoods jsp 


商品 编号 是 主键 ， 不 能 重复 ， 每 个 信息 都 必须 输入 ! 


oodsId goodsName goodsPrice goodsType | 


1 FE 12 日 用 品 | 
2 pa 2500 电器 | 
3 TE 28 食品 | 
4 GER a8 KÆ | 
5 ER (1800 服装 | 
6 书包 78 KA | 
7 ER aT 水 果 | 


616 查询 记录 
代码 模板 addGoods. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset— GBK"pageEncoding— "GBK" % > 
<html> 
<head> 
<title>addGoods. jsp</title> 
</head> 
<body> 
<h4> nA SEER, KAEH, Bp 4 f B. 622 38 À !< /h4—> 
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<form action— "showAllGoods. jsp" method= "post" > 
<table border— "1"7- 
<tr> 
<td> B hf </td> 
<td><input type="text" name— "goodsld" /77 — /td> 
</tr> 


<tr> 

<td> Ri ih APR : </td> 

<td><input type="text" name= "goodsName"/></td> 
</tr> 


<tr> 

<td> Ri sh WK: </td> 

<td><input type="text" name= "goodsPrice"/ ></td> 
</tr> 


<e> 
<td> ili ia KI : </td> 
<td> 
<select name= "goodsType"> 
<option value=" H H i "> H Hi 
"电器 "二 电器 
"食品 "二 食品 
"水 果 " 二 水 果 
<option value 一 "服装 "之 服装 
<option value 一 "文具 "二 文具 
<option value 二 "其 他 "二 其 他 
</select> 
</td> 
</tr> 


<option value= 


<option value 


<option value= 


<tr> 
<td><input type="submit" value— "IJI" ></td> 
<td><input type= "reset" value= "3E "></td> 
</tr> 
</table> 
</form> 
</body> 
</html> 


代码 模板 showAllGoods. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset— GBK"pageEncoding— "GBK" % > 
<%@ page import— "bean. GoodsBean" 967» 
<html> 
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<head> 
<title>showAllGoods. jsp</title> 
</head> 
<body> 
<% 
request. setCharacterEncoding( "GBK" ) ; 
%> 
<jsp:useBean id= "goods" class— "bean. GoodsBean" scope="page"></jsp:useBean> 
—jsp:setProperty property" * " name= "goods" /> 
<% 
goods.addGoods();// 添 加 商品 
%> 
<jsp:getProperty property= "queryResult" name= "goods" /7— !-- 获得 查询 结果 --> 
</body> 
</html> 


代码 模板 GoodsBean. java 如 下 : 


package bean; 
import java. sal. * ; 
public class GoodsBean | 
int goodsId; 
String goodsName; 
double goodsPrice; 
String goodsType; 
StringBuffer queryResult; // 查 询 结果 
public GoodsBean() í 


) 

public int getGoodsId() ( 
return goodsld; 

) 

public void setGoodsId(int goodsId) { 
this. goodsId = goodsld; 

) 

public String getGoodsName() | 
return goodsName; 

) 

public void setGoodsName(String goodsName) ( 
this. goodsName = goodsName; 

) 

public double getGoodsPrice() | 
return goodsPrice; 

) 

public void setGoodsPrice(double goodsPrice) ( 
this. goodsPrice — goodsPrice; 
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) 
public String getGoodsType() í 
return goodsType; 
) 
public void setGoodsType(String goodsType) { 
this. goodsType — goodsType; 
) 
// 添 加 记录 
public void addGoods() | 
Connection con — null; 
Statement st — null; 
try ( 
Class. forName( "oracle. jdbc. driver. OracleDriver") ; 
) catch (ClassNotFoundException e) ( 
e. printStackTrace() ; 
) 
try { 
con= DriverManager. getConnection( "jdbc:oracle: thin: (9 localhost: 1521 :orcl" , 
"system", "system") ; 
st— con. createStatement( ) ; // 创 建 Statement 对 象 
String addSql= "insert into goodsInfo 
values( " + goodsId-- " , '" + goodsName+ " ', "+ goodsPrice+ " , '" +goodsType + "')"; 
st.executeUpdate(addSql) ; // 执 行 insert 语句 
} catch (SQLException e) { 
e. printStackTrace() ; 
) finallyt 
try( 
if(st! =null) 4 
st. close) ; 
) 
if(con! =null) ( 
con. close); 
) 
)catch (SQLException e) ( 
e. printStackTrace() ; 


) 

// 查 询 记录 

public StringBuffer getQueryResult() ( 
queryResult= new StringBuffer() ; 
Connection con 一 null; 
Statement st 一 null; 
ResultSet rs 一 null; 
try { 
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Class. forName( "oracle. jdbc. driver. OracleDriver") ; 
} catch (ClassNotFoundException e) { 
e. printStackTrace() ; 
) 
try ( 
con— DriverManager. getConnection( "jdbc : oracle : thin: (2 localhost: 1521 : orcl" , 


"system" , "system") ; 


st— con. createStatement() ; // 创 建 Statement 对 象 
String selectSgl — "select * from goodsInfo" ; 
rs— st. executeQuery(selectSql) ; // 执 行 select 语句 


queryResult. append(" —table border=1>"); 
queryResult. append(" — tr"); 
queryResult. append( "<th> goodsId— /th7 ") ; 
queryResult. append( "<th> goodsName< /th>"); 
queryResult. append( "<th> goodsPrice< / th") ; 
queryResult. append( "<th> goodsType- / th") ; 
queryResult. append(" —/tr27") ; 
while(rs. next) | 
queryResult. append(" — tr"); 
queryResult. append( "<td>" 4- rs. getString(1) - " —/td77") ; 
queryResult. append( "<td>" 4- rs. getString(2) + " —/td77") ; 
queryResult. append( "<td>" 4- rs. getString(3) + "< /td>>"); 
queryResult. append( "<td>" + rs. getString(4) +"</td>"); 
queryResult. append("</tr>"); 


) 
queryResult. append(" — /table2") ; 
)catch (SQLException e) | 
e. printStackTrace() ; 
) finallyt 
try( 
if(rs! = null) ( 
rs. close() ; 
) 
if(st! =null) ( 
st. close() ; 
) 
if(con! =null) í 
con. close() ; 
) 
}catch (SQLException e) | 
e. printStackTrace() ; 
) 
) 


return queryResult; 
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ResultSet 对 象 自动 维护 指向 其 当前 数据 行 的 游标 。 每 调用 一 次 next() 方 法 ,游标 向 下 
移动 一 行 。 最 初 它 位 于 结果 集 的 第 一 行 之 前 ,因此 第 一 次 调用 next() ,将 把 游标 置 于 第 一 
行 上 ,使 它 成 为 当前 行 。 随 着 每 次 调用 next() ,导致 游标 向 下 移动 一 行 ,按照 从 上 至 下 的 次 
序 获 取 ResultSet 行 , 实 现 顺序 查询 。 

ResultSet 对 象 包含 SQL 请 句 的 执行 结果 。 它 通过 一 套 get 方法 对 这 些 行 中 数据 的 访 
问 , 即 使 用 getXxx 方 法 获得 数据 。get 方法 很 多 如 表 6-2 Bros ,究竟 用 哪 一 个 getXxx() 方 
法 ,由 列 的 数据 类 型 来 决定 。 

通过 表 6-2 列举 的 方法 不 难看 出 ,在 前 面 的 例子 中 ,也 可 以 将 rs. getString(1) 替 换 成 
rs. getString("goodsId") ,将 rs. getString (2) 替 换 成 rs. getString("goodsName") ,将 rs 
. getString(3) 替 换 成 rs. getString(" goodsPrice") ,将 rs. getString (4) EE Ë J, rs. getString 
("goodsType"), 


#62 Result 对 象 的 getXxx() 方 法 


返回 类 型 方法 名 称 返回 类 型 方法 名 称 
byte getByte(int columnlndex) String getByte(String columName) 
date getDate(int columnIndex) date getDate(String columName) 
double getDouble(int columnIndex) double getDouble(String columName) 
float getFloat(int columnlndex) float getFloat(String columName) 
int getlnt(int columnlndex) int getInt(String columName) 
String getString(int columnIndex) String getString( String columName) 


使 用 getXxx 方法 时 ,还 需要 注意 以 下 两 点 : 

(1) 无 论 列 是 何 种 数据 类 型 ,总 可 以 使 用 getString (int columnlndex) 或 getString 
(String columnName) 方 法 获得 列 值 的 字符 串 表 示 。 

(2) 如 果 使 用 getString(int columnIndex) 方 法 查看 一 行 记 录 时 ,不 允许 颠倒 顺序 , 例 
如 ,不 允许 : 


rs. getString(2); 
rs. getString(1) ; 


6.3.3 拓展 训练 


1. 从 “员工 ” 表 的 “姓名 ”字段 找 出 名 字 包 含 “ 玛 丽 ” 的 人 ,下 面 的 select 语句 中 正确 的 
是 ( Js 
A. select * from 员工 where 姓名 like "Yo pH Yo" 
B. select * from 员工 where VE = '% 玛 丽 _' 
C. select * from 员工 where 姓名 like ' 3j 95" 
D. select * from 员工 where W% = 玛丽， 
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2. 下 面 ( ) 不 是 ResultSet 接口 的 方法 。 
A. next() B. getStringO C. backO D. getlnt() 
3. 编写 两 个 JSP 页 面 : inputQuery. jsp 和 showGoods. jsp。 用 户 可 以 在 inputQuery. jsp 页 
面 输入 查询 条 件 后 , 单 击 “查询 按钮。 然后 ,在 showGoods. jsp 页 面 中 显示 符合 查询 条 件 的 
商品 信息 。 在 本 节 任 务 的 beanCGoodsBean. java) 中 添加 一 个 方法 getQueryResultBy() 实 
现 该 题 的 条 件 查询 功能 。 页 面 运行 效果 如 图 6-17 和 图 6-18 所 示 。 


http://localhost:8080/ch06/inputQueryjsp 


http://localbost:8080/ch06/showGoods_jsp 


|goodsId lgoodsName goodsPrice lgoodsType 
a BER 48 KA 
F mx 17 pK 


图 6-17 输入 条 件 图 6-18 符合 查询 条 件 的 记录 


6.4 游 动 查询 


6.4.1 知识 要 点 


有 时 候 需要 结果 集 的 游标 前 后 移动 ,这 时 可 使 用 滚动 结果 集 。 为 了 获得 滚动 结果 集 , 必 
须 先 用 下 面 方法 得 到 一 个 Statement 对 象 : 


Statement st— con. createStatement(int type, int concurrency) ; 


根据 type 和 concurrency 的 取 值 , 当 执行 ResultSet rs— st. executeQuery(String sql) 时 ,会 
返回 不 同类 型 的 结果 集 。 
type 的 取 值 决定 滚动 方式 , 它 可 以 取 如 下 值 : 
。 ResultSet. TYPE_FORWORD_ONLY 一 一 表示 结果 集 只 能 向 下 滚动 。 
* ResultSet. TYPE_SCROLL_INSENSITIVE 一 一 表示 结果 集 可 以 上 下 滚动 , 当 数 据 
库 变 化 时 ,结果 集 不 变 。 
* ResultSet. TYPE_SCROLL_SENSITIVE 一 一 表示 结果 集 可 以 上 下 滚动 , 当 数据 库 
变化 时 ,结果 集 同 步 改变 。 
concurrency 取 值 表示 是 否 可 以 用 结果 集 更 新 数据 库 , 它 的 取 值 是 : 
。 ResultSet. CONCUR_READ_ONLY 一 一 表示 不 能 用 结果 集 更 新 数据 库 的 表 。 
。 ResultSet. CONCUR_UPDATETABLE 一 一 表示 能 用 结果 集 更 新 数据 库 的 表 。 
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6.4.2 技能 操作 


使 用 滚动 结果 集 进 行 游 动 查询 ,具体 任务 如 下 : 编写 一 个 JSP 页 面 randomQuery. jsp» 
查询 goodsInfo 表 中 的 全 部 记录 ,并 将 结果 逆序 输出 ,最 后 单独 输出 第 4 条 记录 。 和 运行 结果 
如 图 6-19 所 示 。 


http://localhost:8080/ch06/randomQueryjsp 
该 表 共 有 6 条 记录 

现在 逆序 输出 记录 ， 

商品 编号 商品 名 称 商品 价格 商品 类 别 


[d TR i7 KE 
6 BE [78 KA 
5 ER [1800 服装 
la GE lag KA 
2 DKI [2500 电器 

n FE hz BAS 
单独 输出 第 4 条 记录 


5 EX 1800 服装 


6-19 ”随机 查询 记录 
代码 模板 randomQuery. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset=GBK"pageEncoding="GBK" %> 
<%@ page import= "java. sql. * "%> 
<html> 
<head> 
<title> randomQuery. jsp / title 
</head> 
<body> 
<% 
Connection con = null; 
Statement st = null; 
ResultSet rs = null; 
try ( 
Class. forName( "oracle. jdbc. driver. OracleDriver") ; 
) catch (ClassNotFoundException e) { 
e. printStackTrace() ; 
) 
try | 
con = DriverManager. getConnection ( " jdbc: oracle: thin: (9 localhost: 1521: orcl", 
"system", system") ; 
// 创 建 st 对 象 ,该 对 象 获得 的 结果 集 和 数据 库 同步 变化 ,但 不 能 用 结果 集 更 新 表 
st = con. createStatement( ResultSet. TYPE_SCROLL_SENSITIVE, 
ResultSet.CONCUR_READ_ONLY); // 返 回 可 滚动 的 结果 集 
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rs = st.executeQuery(" SELECT * FROM goodsInfo"); 
// 将 游标 移动 到 最 后 一 行 

rs. last(); 

// 获 取 最 后 一 行 的 行 号 

int lownumber = rs.getRow(); 
out.print(" 该 表 共 有 " + lownumber + "条 记录 "); 
.print("<<BR 之 现在 逆序 输出 记录 : "); 

out. print("< Table Border=1>"); 
.print("<TR>"); 

.print("  TH2 Hi m A 5 —/ TH"); 

.print("  TH2 Bi AA Fk / TH"); 

.print(" TH HÑ si fff —/ TH"); 

.print(" TH Ñ 8828 9I — /TH2") ; 

out. print("  /TR"); 

// 为 了 逆序 输出 记录 , 需 将 游标 移动 到 最 后 一 行 之 后 


rs.afterLast(); 


ou 


- 


ou 


= 


ou 


ou 


ou 


E 


ou 


E 


while (rs. previous()) { 
out. print(" TR"); 
out. print(" TD >" + rs. getString(1) + "</TD>"); 
out. print(" TD >" + rs. getString(2) + "</TD>"); 
out. print(" TD >" + rs. getString(3) + "</TD>"); 
out. print(" TD >" + rs. getString(4) + "</TD>"); 
out. print("</TR>"); 
) 
out. print(" —/ Table"); 
out. print( "< BR f: Jhih ih 38 4 Aio BRA); 
rs. absolute(4) ; 
out. print(rs. getString(1) + " "); 
out. print(rs. getString(2) + " "); 
out. print(rs. getString(3) + " "); 
out. print(rs. getString(4)) ; 
) catch (SQLException e) { 
e. printStackTrace() ; 
) finally { 
try ( 
if (rs ! — null) ( 
rs.closeO ; 
) 
if (st ! — null) ( 
st. close) ; 
) 
if (con ! — null) ( 
con. close) ; 
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) 
) catch (SQLException e) í 
e. printStackTrace() ; 
) 
) 
> 


</body> 
</html> 


游 动 查询 经 常用 到 ResultSet 类 的 下 述 方法 : 


public boolean absolute(int row) 一 一 将 游标 移 到 参数 row 指定 的 行 。 如 果 row 取 
负 值 ,就 是 倒数 的 行 ,如 一 1 表示 最 后 一 行 。 当 移 到 最 后 一 行 之 后 或 第 一 行 之 前 时 ， 
该 方法 返回 false。 

public void afterLast() 一 一 将 游标 移 到 结果 集 的 最 后 一 行 之 后 。 

public void beforeFirst() 一 一 将 游标 移 到 结果 集 的 第 一 行 之 前 。 

public void first() 一 一 将 游标 移 到 结果 集 的 第 一 行 。 

public int getRow() 一 一 得 到 当前 游标 所 指定 的 行 号 ,如 果 没 有 行 , 则 返回 0。 
public boolean isAfterLast() 一 一 判断 游标 是 不 是 在 结果 集 的 最 后 一 行 之 后 。 
public boolean isBeforeFirst() 一 一 判断 游标 是 不 是 在 结果 集 的 第 一 行 之 前 。 

public void last() 一 一 将 游标 移 到 结果 集 的 最 后 一 行 。 

public boolean previous() 一 一 将 游标 向 上 移动 (和 next 方法 相反 ), 当 移动 结果 集 
的 第 一 行 之 前 时 返回 false。 


6.4.3 拓展 训练 


L 


2. 


下 述 选 项 中 不 属于 JDBC 基本 功能 的 是 ( Js 

A. 与 数据 库 建 立 连接 B. 提交 SQL 语句 

C. 处 理 查询 结果 D. 数据 库 维 护 管理 

编写 一 个 JSP 页 面 practice6 4. jsp. ifi] goodsInfo 表 中 的 记录 ,并 逆序 输出 偶数 行 


的 记录 。 运 行 结果 如 图 6-20 所 示 。 


http://localhost:8080/ch06/practice6 4jsp 

该 表 共 有 6 条 记录 

现在 送 序 输出 偶数 行 的 记录 ， 

行 号 商品 编号 商品 名 称 商品 价格 商品 类 别 
et m == 17 = 

at 5 LEX [1800 RE 

AT [2 DRI 2500 | 电器 


图 6-20 ” 道 序 输出 偶数 行 的 记录 
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6.5 使 用 PreparedStatement 操作 数据 


与 Statement 语句 一 样 ,PreparedStatement 也 同样 可 以 完成 向 数据 库 发 送 SQL 语句 ， 
获取 数据 库 操作 结果 的 功能 。 但 是 Statement 对 象 在 每 次 执行 SQL 语句 时 都 将 该 语句 传 
送 给 数据 库 , 然 后 数据 库 解 释 器 负责 将 SQL 语句 转换 成 内 部 命令 ,并 执行 该 命令 ,完成 相应 
的 数据 库 操 作 。 基 于 这 种 机 制 ,每 次 向 数据 库 发 送 一 条 SQL 语句 时 ,都 要 先 转化 成 内 部 命 
令 ,如果 不 断 地 执行 程序 ,就 会 加 重 解释 器 的 负担 ,影响 执行 的 速度 。 

而 PreparedStatement 对 象 将 SQL 语句 传 给 数据 库 进 行 预 编译 ,以 后 需要 执行 同一 条 
语句 时 就 不 需要 再 重新 编译 ,直接 执行 就 可 以 了 ,这样 就 大 大 提高 了 数据 库 的 执行 速度 。 


6.5.1 知识 要 点 


可 以 使 用 Connection 的 对 象 con 调用 prepareStatement(String sql) 方 法 对 参数 sql 指 
定 的 SQL 语句 进行 预先 编译 ,生成 数据 库 的 底层 命令 , 并 将 该 命令 封装 在 
PreparedStatement 对 象 中 。 对 于 SQL 请 句 中 会 变动 的 部 分 ,可 以 使 用 通配符 "?” 代 替 。 
例如 : 


然后 使 用 对 应 的 setXxx(int parameterIndex, xxx value) 方 法 指定 “?” 代 表 的 值 .其 中 参数 
parameterIndex 用 来 表示 SQL 语句 中 从 左 到 右 的 第 parameterIndex 个 通配符 号 ,value 代 
表 该 通配符 所 代表 的 具体 值 。 例 如 : 

ps. setInt(1,9); 

ps.setString(2, "手机 "); 

ps. setDouble(3, 1900.8); 

ps.setString(4, "通信 "); 
若 要 让 SQL 语句 执行 生效 , 需 使 用 PreparedStatement 的 对 象 ps 调用 executeUpdate O Jf 
法 。 如 果 是 查询 的 话 ,ps 就 调用 executeQuery() 方 法 ,并 返回 ResultSet 对 象 。 


6.5.2 技能 操作 


使 用 PreparedStatement 预 处 理 语句 对 象 操作 数 据 库 中 的 表 。 具 体 任务 如 下 : 

编写 两 个 JSP 页 面 一 一 inputPrepareGoods. jsp 和 showPrepareGoods. jsp。 用 户 可 以 
在 inputPrepareGoods. jsp 页 面 中 输入 信息 后 , 单 击 “ 添 加 ”按钮 把 信息 添加 到 goodsInfo 表 
中 。 然 后 ,在 showPrepareGoods. jsp 页 面 中 显示 所 有 商品 信息 。 在 该 任务 中 需要 编写 一 个 
bean(UsePrepare. java) ,bean 中 使 用 预 处 理 语 句 向 goodsInfo 表 中 添加 记录 。 页 面 运行 效 
果 如 图 6-21 和 图 6-22 所 示 。 
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http.//localhost:8080/chO6/inputPrepareGoodsjsp be fiocoliost noe oec repareticode gu 
商品 编号 是 主键 不 能 重复 .每 个 信息 都 必须 输入 1 goodsId goodsName goodsPrice goodsType 
1 FE o am BAS 
商品 编号 : 2 p 2500 电器 
商品 名 称 : [xn ] s BUR as HR 
Batik: s — | 5 [ER 1800 服装 
KEN las Ç — 6 ma 78 ZA 
— T X i ** 
Ha Ma 10 BEM — 299.99 Ld 


图 6-21 使 用 预 处 理 添加 商品 图 6-22 ”使 用 预 处 理 查询 商品 


代码 模板 inputPrepareGoods. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK " 
"GBK"%> 
<html> 
<head> 
<title> f E BUE BEB A] — title> 
</head> 
<body bgcolor=" LightYellow" > 
<h4> Wi ih a 5 Je dc E A BE GRE BA fei BAR LB À !</h4— 
<form action= "showPrepareGoods. jsp" method= "post" > 
<table border— "1"> 
«tu 
二 td 二 商品 编号 :二 /td 二 
<td><input type="text" name= "goodslId" /7 — /td 
</tr> 


<tr> 

<td> Hi ih # fr: </td> 

<td><input type="text" name= "goodsName"/> </td> 
</tr> 


<tr> 
<td> R ñ M : </td> 


<td><input type="text" name= "goodsPrice"/></td> 
</tr> 


<u> 
<td> ili b KI : </td> 
<td> 
<select name= "goodsType" > 
<option value=" H H "> H Hh 
<option valus 电器 "二 电器 
<option value 二 "食品 "二 食品 


pageEncoding = 
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<option value= "7K "7k JE 
<option value=" RE ">R 
<option value=" XA "> XA 
<option value 一 "其 他 "二 其 他 
</select> 
</td> 
</tr> 


<tr> 
<td><input type="submit" value= "illl" ></td> 
<td><input type= "reset" value— "3E "></td> 
</tr> 
</table> 
</form> 
</body> 
</html> 


代码 模板 showPrepareGoods. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset=GBK"pageEncoding="GBK" %> 


<%@ page import= "bean. UsePrepare" 267» 
<html> 
<head> 
<title> fl H MERE </title> 
</head> 
<body> 
<% 
request. setCharacterEncoding( " GBK") ; 
5n 


< jsp: useBean id =" prepareGoods" class =" bean. UsePrepare" scope =" page" > — /jsp: 


useBean > 
<jsp:setProperty property= " * " name="prepareGoods"/> 
<% 


prepareGoods. addGoods() ; // 添 加 商品 
> 
<jsp:getProperty property= "queryResult" name= "prepareGoods" />><!-- 获得 查询 结果 --> 
</body> 
</html> 


代码 模板 UsePrepare. java 如 下 : 


package bean; 
import java. sal. * ; 
public class UsePrepare ( 
int goodsld; 
String goodsName; 
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double goodsPrice; 

String goodsType; 

StringBuffer queryResult; // 查 询 所 有 商品 结果 
public UsePrepare() ( 


) 
public int getGoodsId() í 
return goodsld ; 
) 
public void setGoodsId(int goodsId) í 
this.goodsId = goodsld; 
) 
public String getGoodsName() | 
return goodsName; 
) 
public void setGoodsName(String goodsName) { 
this. goodsName — goodsName; 
) 
public double getGoodsPrice() { 
return goodsPrice; 
) 
public void setGoodsPrice(double goodsPrice) ( 
this. goodsPrice — goodsPrice; 
) 
public String getGoodsType() | 
return goodsType; 
) 
public void setGoodsType(String goodsType) í 
this. goodsType = goodsType; 
) 
// 添 加 商品 
public void addGoods(){ 
Connection con = null; 
PreparedStatement ps = null; 
try { 
Class. forName( "oracle. jdbc. driver. OracleDriver") ; 
) catch (ClassNotFoundException e) í 
e. printStackTrace() ; 
) 
try ( 
con= DriverManager. getConnection( "jdbc : oracle: thin: (2 localhost : 1521 :orcl" , 
"system", system") ; 
ps con. prepareStatement( "insert into goodsInfo values(?, ?,?, ?) ") ; 
ps.setInt(1, goodsld) ; / / ps 调用 set 方法 指定 第 1 个 通配符 的 值 
ps.setInt(1, goodsld) ; / / ps 调用 set 方法 指定 第 2 个 通配符 的 值 


ps. setInt(1，goodsId) ; 
ps.setString(4, goodsType) ; 
ps.executeUpdate() ; 
)catch (SQLException e) | 
e. printStackTrace() ; 
) finallyt 
try( 
if(ps! = null) { 
ps. close() ; 
} 
ifCcon! — null) ( 
con.close() ; 
) 
)catch (SQLException e) | 
e. printStackTrace() ; 
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) 
// 获 得 所 有 商品 信息 
public StringBuffer getQueryResult() ( 


queryResult— new StringBuffer() ; 
Connection con — null; 
PreparedStatement ps — null; 
ResultSet rs— null; 

try ( 


Class. forName( "oracle. jdbc. driver. OracleDriver") ; 


) catch (ClassNotFoundException e) { 
e. printStackTrace() ; 

) 

try | 


//ps 调用 set 方法 指定 第 3 个 通配符 的 值 
//ps 调用 set 方法 指定 第 4 个 通配符 的 值 


con= DriverManager. getConnection( "jdbc:oracle: thin: @localhost:1521 :orcl", 


"system","system"); 


ps 一 con. prepareStatement("select * from goodsInfo") ; 


rs— ps. executeQuery() ; 


queryResult. append(" table border— 177"); 


queryResult. append(" — tr") ; 


queryResult. append(" th goodsId— /th>"); 
queryResult. append(" th» goodsName- /th>2>") ; 
queryResult. append( "<th> goodsPrice— / th") ; 
queryResult. append("<th> goodsType- / th") ; 


queryResult. append(" — /tr7"); 


while(rs. next()) ( 
queryResult. append(" — tr"); 


queryResult. append( "<td>" 
queryResult. append( "<td>" 


Hrs. getString(1)+"</td>"); 
Hrs. getString(2)+"</td>"); 
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queryResult.append( "< td>>"+rs.getString(3) + "< /td»") ; 
queryResult. append( "<td>" 4- rs. getString(4) + "< /td>2>") ; 
queryResult. append(" — /tr»"); 
) 
queryResult. append(" — / table") ; 
)catch (SQLException e) | 
e. printStackTrace() ; 
) finallyt 
try{ 
if(rs! — null) ( 
rs.close() ; 
) 
ifCps! — nulb ( 
ps. close(); 
} 


if(con! =null) ( 


con.close() ; 


) 
)catch (SQLException e) í 
e. printStackTrace() ; 
) 
) 


return queryResult; 


) 


通过 之 前 的 学 习 , 我 们 已 经 知道 Statement 在 执行 executeQuery (String sql), 
executeUpdate(String sql) 等 方法 时 ,如 果 SQL 请 句 有 些 部 分 是 动态 的 数据 ,必须 使 用 “十 ” 
连 字 符 组 成 完整 的 SQL 请 句 , 十 分 不 便 。 例 如 6. 3 节 中 的 任务 在 添加 商品 时 ,必须 按 如 下 
方式 组 成 SQL 语句 : 


"4 ("a 


String addSql= "insert into goodsInfo values(" + goodsId + ", '" + goodsName + " ', " + goodsPrice + 

",'™+goodsType+"')"; 

st. executeUpdate(addSql) ; 

PreparedStatement 对 象 被 称 为 预 处 理 语句 对 象 ,现在 使 用 预 处 理 语句 不 仅 提高 了 数据 
库 的 访问 效率 ,而 且 方 便 了 程序 的 编写 。 预 处 理 语句 对 象 调用 executeUpdate CO 和 
executeQuery() 方 法 时 不 需要 传递 参数 。 例 如 : 


int i= ps. executeUpdate() ; 
或 


ResultSet rs= ps. executeQuery() ; 
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6.5.3 拓展 训练 


编写 两 个 JSP 页面: inputPrepareQuery. jsp 和 showPrepareBy. jsp。 用 户 可 以 在 页 面 
inputPrepareQuery. jsp 中 输入 查询 条 件 后 , 单 击 “查询 按钮。 然后 ,在 showPrepareBy. jsp 
页 面 中 显示 符合 查询 条 件 的 商品 信息 。 在 本 节 任 务 的 bean(UsePrepare. java) 中 添加 一 
方法 getQueryPrepareResultBy(O) 实 现 本 题 的 条 件 查询 功能 (使 用 预 处 理 语句 实现 查询 ) 。 
页 面 运行 效果 如 图 6-23 和 图 6-24 所 示 o 


http://localhosti8080/ch06/inputPrepareQueryjsp 
输入 查询 条 件 ! 


asm. C 
| 商品 价格 大 于 : [so 


|'pulocalbost8080/choe/showPrepareByjsp 


mam: jgoodsId goodsName goodsPrice goodsType 
[sm XT Ë PRIE [2500 电器 
图 6-23 使 用 预 处 理 条 件 查询 6-24 符合 查询 条 件 的 记录 


6.6 访问 Excel 电子 表格 


6.6.1 知识 要 点 


当 需 要 对 Excel 电子 表格 的 内 容 进行 增加 、 修 改 、 查 询 和 删除 时 ,可 以 使 用 JDBC-ODBC 
桥接 器 的 方式 访问 Excel 电子 表格 。 但 访问 Excel 电子 表格 和 访问 关系 数据 库 有 所 不 同 。 
访问 Excel 电子 表格 要 经 过 以 下 两 个 步骤 。 

1. 创建 Excel 电子 表格 

使 用 Microsoft Office Excel 2007 在 物理 硬盘 “D:\” 根 目录 下 创建 一 个 名 为 bookInfo 
. xlsx 的 电子 表格 ,该 Excel 表格 中 有 个 名 为 bookDetail 的 工作 表 , 具 体内 容 如 图 6-25 所 示 。 


lal. F 

作者 出 版 社 出 版 日 期 ” 定价 

9787121 CABAK 王 大 川 , 王 小 川 中国 第 一 出 版 社 2000/9/10 150.5 

3 | 9787122 ，C 语 言 程序 设计 李小萌 明珠 出 版 社 2012/3/1 | 132.0 

4 | 9787123 ， 网 页 设计 与 制作 ATA 编程 出 版 社 2012/6/1  ¥29.5 

5 | 9787124 — JavaWeb 开 发 张 月 月 武功 秘籍 出 版 社 _2014/8/20 | 139.5 

6 | 9787125 ， 办 公 自 动 化 SAI 网 络 时 代 出 版 社  2015/1/5 ^ 328.5 
7 


图 6-25 student. xlsx 电子 表格 
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2. 创建 数据 源 

创建 Excel 电子 表格 之 后 ,就 可 以 按照 6. 1 节 中 介绍 的 方法 为 它 创 建 数 据 源 了 。 首 先 
为 数据 源 选择 的 驱动 程序 是 Microsoft Excel Driver( * . xls, * . xlsx, * . xlsm, * . xlsb) . fA 
后 在 如 图 6-26 所 示 的 对 话 框 中 将 数据 源 命 名 为 bookSource, 接 下 来 单 击 “ 选 择 工 作 短 ?按钮 
选择 上 一 步 中 创建 的 *D:\bookInfo. xlsx? 工 作 短 。 还 需要 尤其 注意 的 是 ,如 果 想 通过 
JDBC-ODBC 修改 Excel 电子 表格 ,在 设置 数据 源 时 , 单 击 “ 选 项 ”按钮 ,将 “只 读 ” 属 性 设置 为 
未 选中 状态 。 


D: VookIn£o. xlsx 
LH 使 用 当前 目录 Q0 
驱动 程序 
Bae: s 


图 6-26 创建 数据 源 
需要 注意 的 是 ,访问 电子 表格 时 ,把 其 中 的 工作 表 看 成 数据 库 中 的 表 , 如 bookInfo. xlsx 


中 的 bookDetail 工作 表 。 使 用 SQL 语句 对 工作 表 中 的 数据 进行 增加 、 删 除 、 修 改 和 查询 时 ， 
要 在 表 名 前 加 “[”, 在 表 名 后 加 “$ ]”, 如 select * from [bookDetail$ ]. 


6.6.2 技能 操作 


使 用 JDBC-ODBC 桥接 器 的 方式 访问 Excel 电子 表格 。 具 体 任务 如 下 : 
编写 一 个 JSP 页 面 readExcel. jsp, 在 该 页 面 的 Java 程序 片 中 首先 增加 一 条 记录 到 
studentScore 工作 表 中 ,然后 修改 了 某 条 记录 ,最 后 查询 全 部 记录 。 页 面 运行 效果 如 图 6-27 所 示 。 


http://localhost:8080/ch06/operateExceljsp 


图 6-27 增加 、 修 改 和 查询 Excel 表 的 记录 
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代码 模板 operateExcel. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<%@ page import— "java. sql. * "% > 
<html> 
<head> 
<title>operateExcel. jsp</title> 
</head> 
<body bgcolor= "lightblue" > 
<% 
Connection con = null; 
Statement st = null; 
ResultSet rs = null; 
try { 
Class. forName( "sun. jdbc. odbc. JdbcOdbcDriver" ) ; 
} catch (ClassNotFoundException e) { 
e. printStackTrace() ; 
) 


try ( 
con = DriverManager. getConnection( "jdbc:odbc: bookSource" , "", ""); 
st— con. createStatement( ) ; 
// 添 加 记录 
st. executeUpdate( "insert into [bookDetail $ ] 
values('9787126', ' 计 算 机 网 络 ', ' 李 三 ',' 启 明星 出 版 社 ', '2015/4/1', '¥ 30.09"); 
// 修 改 记录 


st.executeUpdate( "update [bookDetail $ ] 
set 出 版 社 一 ' 中 国 第 一 出 版 社 ' where [8 48 = 978712211); 
// 查 询 全 部 记录 
rs=st. executeQuery( "select * from [bookDetail $ ]") ; 
out. print("— table border=1>"); 
out. print(" tr"); 
out. print(" th A BAE </th>"); 
out. print(" <th> A 4 fk -—/th"); 
out. print(" th [EAE — /th»"); 
out. print(" <th> Hi dE / th") ; 
out. print(" th tH f H Ej — /th"); 
out. print(" th Æ fft — /th2"); 
out. print("  /tr7") ; 
while(rs. next()) ( 
out. print(" — tr"); 
out. print( "<td>" 4-rs. getString(1) +"</td>"); 
out. print( "<td>" rs. getString( 2) - "<< /td77"); 
out. print( "<td>" trs. getString(3) +"</td>" 
out. print( "<td>" + rs. getString( 4) T "< /td77"); 
out. print( "<td>" + rs. getString( 5) - "  /td77"); 
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out. print(" « td>" 4- rs. getString(6) +"</td>"); 
out. print("  /tr7"); 
) 
out. print(" — /table27") ; 
) catch (SQLException e) { 
e. printStackTrace() ; 
) finally( 
try( 
if(rs! =null) ( 
rs.close(); 
) 
ifCst! =null) í 
st.close() ; 
) 
if(con! — nulD ( 
con.close() ; 
) 
)catch (SQLException e) ( 
e. printStackTrace() ; 
) 
) 
n 
</body> 
</html> 


说 明 : — Excel 电子 表格 可 以 有 多 个 工作 表 , 我 们 使 用 JDBC-ODBC 可 以 访问 该 电子 
表格 中 的 任何 一 个 工作 表 , 就 像 访 问 一 个 数据 库 中 的 任意 一 张 表 一 样 。 


6.6.3 拓展 训练 


在 bookInfo. xlsx 电子 表格 中 新 建 一 个 工作 表 bookFare, 详 细 内 容 如 图 6-28 所 示 ; 编写 一 
个 JSP 页 面 practiceExcel. jsp, 在 该 页 面 中 显示 bookFare 表 中 的 所 有 记录 ,如 图 6-29 所 示 。 


FEF = 
J20 X 
D 
1 图书 编号 图 书 名 称 作者 应 付 稿酬 
2 | 9787121 2# 项 目 开 发 TAJI, EMI 1500.50 
3 | 9787122 吾 言 程序 设 李小萌 X600. 60 
| 4 | 9787123 页 设计 与 制 PA 4700. 70 
5 | 9787124 avaWeb 开 和 张 月 月 X800. 80 
6 | 9787125 办 公 自 动 化 孙 达 江 1900. 90 
7 | 9787126 计算 机 网 络 李 三 11, 000. 00 
8 | 9787126 计算 机 网 络 三 500. 00 
i] hookFare. D [ 


6-28 工作 表 bookFare 
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http;//localhost:8080/ch06/bookFare jsp. 


图 6-29  practiceExcel. jsp 页 面 效果 


6.7 使 用 数据 库 连 接 池 


6.7.1 知识 要 点 


和 数据 库 建 立 连接 是 一 个 费时 的 活动 ,每 次 都 要 花费 一 定 的 时 间 ,而 且 系 统 还 要 分 配 内 
存 资源 。 这 个 时 间 对 于 一 次 或 几 次 数据 库 操作 ,或 许 感觉 不 出 系统 有 多 大 的 开销 。 可 是 对 
于 大 型 电子 商务 网 站 同时 有 几 百 人 甚至 几 千 人 频繁 地 进行 数据 库 连接 操作 的 情况 ,势必 占 
用 很 多 的 系统 资源 ,网 站 的 响应 速度 必定 下 降 ,严重 时 甚至 会 造成 服务 器 的 骨 溃 。 因 此 , 合 
理 地 建立 数据 库 连接 是 非常 重要 的 。 

数据 库 连 接 池 的 基本 思想 是 : 为 数据 库 连 接 建 立 一 个 “缓冲 池 ”。 预 先 在 “缓冲 池 ” 中 放 
入 一 定数 量 的 连接 , 当 需 要 建立 数据 库 连 接 时 ,只 需 从 “缓冲 池 ” 中 取出 一 个 ,使 用 完毕 之 后 
再 放 回 去 。 可 以 通过 设 定 连 接 池 最 大 连接 数 来 防止 系统 无 限度 地 与 数据 库 连 接 。 更 为 重要 
的 是 ,通过 连接 池 的 管理 机 制 监视 数据 库 连 接 的 数量 及 使 用 情况 ,为 系统 开发 .测试 和 性 能 
调整 提供 依据 。 其 工作 原理 如 图 6-30 所 示 。 


Servlet 或 JSP 


请 求 
用 户 1 [一 (线程 1 


请 求 
用 户 2 线程 2 


取得 连接 


放 回 连接 


6-30 连接 池 的 原理 
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6.7.2 技能 操作 


使 用 连接 池 连 接 数据 库 。 具 体 任务 如 下 : 

编写 一 个 JSP 页 面 conPool. jsp, 在 该 页 面 中 使 用 scope 为 application 的 bean (由 
ConnectionPool 类 负责 创建 )。 该 bean 创建 时 ,将 
建立 一 定数 量 的 连接 对 象 。 因 此 ,所 有 的 用 户 将 共 
享 这 些 连 接 对 象 。 在 JSP 页 面 中 使 用 bean 获得 一 MERS KEN Wai WISH 
个 连接 对 象 ,然后 使 用 该 连接 对 象 访问 数据 库 中 的 后 EX f1800 RE 
goodsInfo 表 ( 查 询 出 商品 价格 大 于 500 的 商品 )。 
页 面 运行 效果 如 图 6-31 所 示 。 6-31 使 用 连接 池 连 接 数 据 库 


代码 模板 ConnectionPool. java 如 下 : 


http://localhost8080/ch06/conPooljsp 


package db. connection. pool; 
import java. sal. * ; 
import java. util. ArrayList; 
public class ConnectionPool { 
/ / f£ i Connection 对 象 的 数组 ,数组 被 看 成 连接 池 
ArrayList<Connection > list=new ArrayList<Connection> () ; 
// 构 造 方法 ,创建 15 个 连接 对 象 , 放 到 连接 池 中 
public ConnectionPool( ) í 
try { 
Class. forName( "oracle. jdbc. driver. OracleDriver") ; 
) catch (ClassNotFoundException e) | 
e. printStackTrace() ; 
) 
for(int i=0;i<15;i++){ 
try ( 
Connection con= DriverManager. getConnection( "jdbc: oracle: thin: (9 localhost : 1521 : 
orcl" , "system", "system") ; 
list. add(con) ; 
) catch (SQLException e) ( 
// TODO Auto-generated catch block 
e. printStackTrace() ; 


) 
) 
// 从 连接 池 中 取出 一 个 连接 对 象 
public synchronized Connection getOneCon() ( 
if(list. size() 70) í 
// 删 除数 组 的 第 一 个 元 素 , 并 返回 该 元 素 中 的 连接 对 象 
return list. remove(0) ; 
Jelse( 
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// 连 接 对 象 被 用 完 


return null; 


} 

// 把 连接 对 象 放 回 连接 池 中 

public synchronized void releaseCon(Connection con) ( 
list. add(con) ; 


) 
代码 模板 conPool. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GB] 
"GBK"%> 

<%@ page import— "java. sql. * "%> 

<% (à page import— "db. connection. pool. * "547 

<html> 

<head> 

«ie fl H3 Benb Ye Hei E </title> 

</head> 

<jsp:useBean id= "conpool" class= "db. connection. pool. ConnectionPool" scope= "application" /> 
<body bgcolor=" AliceBlue" > 

<% 


Connection con=null; 


pageEncoding = 


Statement st=null; 
ResultSet rs— null; 
try{ 
// 使 用 conpool 对 象 调用 getOneCon 方法 从 连接 池 中 获得 一 个 连接 对 象 
con 一 conpool. getOneCon() ; 
if(con 一 一 null){ 
out. print(" 人 数 过 多 , 稍 后 访问 "); 
return; 
) 
st= con. createStatement() ; 
rs— st. executeQuery("select * from goodsInfo where goodsPrice- 500") ; 
out. print( "< table border—17»"); 
out. print(" tr"); 
out. print(" th fj 56 4 5 — / th") ; 
out. print(" th fj ih 44 Ff — / th") ; 
out. print(" <th> fli Sh fff — / tho") ; 
out. print(" th fli 2E SI — / th") ; 
out. print( "< /tr2>"); 
while(rs.next())( 
out. print(" tr") ; 
out. print( "<td>" +rs. getString(1) + "< /td>"); 
out. print("< td>" 4-rs. getString(2) + "< /td7»"); 
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out. print(" td» "++rs. getString(3) +"</td>"); 
out. print(" td" +rs. getString( 4) +"</td>"); 
out. print(" — /tr7") ; 
) 
out. print("— /table>"); 
} catch( SQLException e) ( 
e. printStackTrace() ; 
}finally{ 
try{ 
if(rs! — null) 4 
rs.close() ; 
) 
if(st!=null) í 
st. close() ; 
) 
if(con! — nul í 
/ / f FA conpool 对 象 调用 releaseCon 方法 把 连接 对 象 放 回 连接 池 中 
conpool. releaseCon(con) ; 
) 
) catch( SQLException e) í 
e. printStackTrace() ; 
) 
) 
5» 
</body> 
</html> 


我 们 再 打开 一 个 新 的 浏览 器 窗口 运行 conPool. jsp 页 面 时 ,会 发 现 这 一 次 访问 的 速度 
要 比 第 一 次 快 得 多 ,而且 也 比 前 面 介 绍 的 访问 JSP 页 面 的 速度 要 快 ,这 是 因为 在 第 一 次 访 
问 时 连接 池 中 没有 可 用 连接 ,因此 页 面 要 等 待 创建 一 个 新 的 连接 ,但 是 在 第 二 次 访问 时 连接 
池 中 就 有 一 个 可 用 连接 了 ,可 以 直接 使 用 这 个 连接 来 访问 数据 库 。 


6.7.3 拓展 训练 
编写 一 个 JSP 页 面 praticePool. jsp, 在 该 页 面 中 使 用 和 技能 操作 中 同样 的 bean 获得 一 
个 数据 库 连 接 对 象 ,然后 使 用 该 连接 对 象 查询 goodsInfo 表 中 的 全 部 记录 。 
6.8 其 他 常见 数据 库 的 连接 


6.8.1 知识 要 点 


使 用 JDBC-ODBC 桥接 器 的 方式 连接 不 同类 型 数据 库 的 程序 流程 是 类 似 的 ,只 是 在 为 
ODBC 数据 源 选择 驱动 程序 时 ,选择 对 应 的 驱动 程序 即 可 ,但 有 的 数据 库 ODBC 不 支持 ,如 
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MySQL 数据 库 。 

使 用 纯 Java 数据 库 驱 动 程序 连接 不 同类 型 数据 库 的 程序 流程 和 框架 是 基本 相同 的 , 需 
要 重点 关注 的 是 ,在 连接 各 数据 库 时 驱动 程序 加 载 部 分 的 代码 和 连接 部 分 的 代码 。 下 面 分 
别 介 绍 直 接 加 载 纯 Java 数据 库 驱动 程序 连接 SQL Server 2005 和 MySQL 5.5。 这 里 假定 
要 访问 的 数据 库 名 为 mydatabase。 

1. 连接 SQL Server 2005 

(1) 获取 纯 Java 数据 库 驱 动 程序 。 

可 以 登录 微软 的 官方 网 站 http://www. microsoft. com 下 载 Microsoft SQL Server 
2005 JDBC Driver 1.2, 解 奈 Microsoft SQL Server 2005 jdbc driverl. 2. exe 后 得 到 sqljdbc. jar 
文件 。 然 后 ,把 sqljdbc. jar 文件 复制 到 Web 应 用 程序 的 /WEB-INFVlib 文件 夹 中 。 

(2) 加 载 驱动 程序 。 


Class. forName( "com. microsoft. sqlserver. jdbc. SQLServerDriver") ; 
(3) 建立 连接 。 


Connection con= 
DriverManager. getConnection ( " jdbc: sqlserver://localhost: 1433; DatabaseName = 
mydatabase ", 
"用 户 名 "，" 密 码 "); 

2. 连接 MySQL 5. 5 

(1) 获取 纯 Java 数据 库 驱 动 程序 。 

可 以 登录 MySQL 的 官方 网 站 http://www. mysql. com 下 载 驱动 程序 。 这 里 下 载 的 
是 mysql-connector-java-5. 0. 6-bin. jar。 然 后 ,把 mysql-connector-java-5. 0. 6-bin. jar 文件 
复制 到 Web 应 用 程序 的 /WEB-INFVlib 文件 夹 中 。 

(2) 加 载 驱动 程序 。 


Class. forName(" com. mysql. jdbc. Driver") ; 
(3) 建立 连接 。 


Connection con= 
DriverManager. getConnection( "jdbc:mysql: / /localhost : 3306/ mydatabase" , "用 户 名 ", "E 3"); 


6.8.2 技能 操作 


使 用 纯 Java 数据 库 驱 动 程序 连接 其 他 常见 数据 库 , 如 MySQL。 具 体 任务 如 下 : 
。 安装 MySQL 5.5, 

。 创建 数据 库 和 数据 表 。 

。 连接 数据 库 并 操作 数据 表 。 
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1. 安装 MySQL 5.5 

登录 官网 下 载 MySQL 5. 5 安装 程序 mysql-5. 5. 19-win32. msi, 按照 默认 安装 即 可 。 
用 户 名 和 密码 都 设置 为 root。 

2. 创建 数据 库 和 数据 表 

MySQL 5.5 RIJA XE £E "JF li" >“ FREI" MySQL MySQL Server 5. 5>MySQL 
5. 5 Command Line Client 命令 。 在 出 现 的 界面 中 输入 默认 密码 : root', 成 功 启动 MySQL 
监视 器 后 ,在 MS-DOS 窗口 中 出 现 “mysql 二 ”字样 ,如 图 6-32 所 示 。 


Command Line Client 


Welcome to the MySQL monitor. Commands end with ; or Ng. 
[Your MySQL connection id is 4 
[Server version: 5.5.19 MySQL Community Server (GPL) 


[Copyright <c> 2000, 2011, Oracle and/or its affiliates. 811 righ 


or ’\h’ for help. Type '*c' to clear the current input statenent. 


Kd 6-32 启动 MySQL 监视 器 
成 功 启动 MySQL 监视 器 后 ,就 可 以 使 用 如 下 命令 : 
create database mydatabase; 


创建 数据 库 mydatabase, 为 了 在 mydatabase 数据 库 中 创建 表 , 必 须 先进 入 该 数据 库 , 命 令 
如 下 : 


use mydatabase 


进入 数据 库 mydatabase 的 操作 如 图 6-33 所 示 musql? use nydatabase 
进入 数据 库 后 ,就 可 以 创建 表 employee。 创 建 表 的 命 MAARE ni 
令 如 下 : 


musq1> 


- 
create table employee ( 图 6-33 进入 数据 库 
empNo varchar(10) not null, 
name varchar(30) not null, 
salary float not null, 
primary key(empNo) 
); 
insert into employee values( '001', 'zhou', 1200); 
insert into employee values( '002', 'wu',2500); 
insert into employee values( '003', 'zheng', 2800 ) ; 


insert into employee values( '004', 'wang', 4800) ; 
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insert into employee values( '005', 'zhao', 1800) ; 

insert into employee values( '006', 'qian', 7800) ; 

commit; 

3. 连接 数据 库 并 操作 数据 表 

编写 一 个 JSP 页 面 useMySQL. jsp, 在 该 页 面 的 Java 程序 片 中 连接 MySQL 数据 库 ,并 
从 employee 表 中 查询 出 工资 高 于 2000 元 的 雇员 。 页 面 运行 效果 如 图 6-34 所 示 。 


图 6-34 工资 高 于 2000 元 的 雇员 
代码 模板 useMySQL. jsp 如 下 : 


— 49 page language= "java" contentType= "text/html; charset=GBK"pageEncoding="GBK" %> 
<%@ page import— "java. sql. * " %> 
<html> 
<head> 
<title> useMySQL. jsp</title> 
</head> 
<body bgcolor= "lightblue" > 

<% 

Connection con = null; 


Statement st = null; 
ResultSet rs = null; 
try ( 
Class. forName("com. mysql. jdbc. Driver"); //MWš& MySQL 的 Java 驱动 
} catch (ClassNotFoundException e) { 
e. printStackTrace() ; 
) 
try ( 
//*j MySQL 建立 连接 ,数据 库 名 为 mydatabase, 用 户 名 和 密码 都 为 root 
con= DriverManager. getConnection( "jdbc:mysql:/ /localhost:3306/mydatabase", "root", "root"); 
st= con. createStatement() ; 
rs—st.executeQuery("select * from employee where salary>2000"); 
out. print("— table border— 177"); 
out. print(" tr"); 
out. print(" th Jg fA 4j S — /th>2>") ; 
out. print(" th ë B Ed < /th>2>") ; 
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out. print(" th T VE /th"); 
out. print("  /tr7"); 
while(rs. next) ( 
out. print(" tr"); 
out. print( "<td>" «rs. getString( 1) 4- " /td»"); 
out. print( "<td>" d- rs. getString(2)+"</td>"); 
out. print( "<td>" 4-rs. getString( 3) - "  /td7»") ; 
out. print(" — /tr7"); 
) 
out. print(" — /table-") ; 
) catch (SQLException e) { 
e. printStackTrace() ; 
) finally( 
try{ 
if(rs! =null) í 
rs.close(); 
) 
if(st! =null) ( 
st. close) ; 
) 
if(con! =null) ( 
con. close() ; 
) 
)catch (SQLException e) | 
e. printStackTrace() ; 


) 

> 
</body> 
</html> 


6.8.3 拓展 训练 


1. 微软 公司 提供 的 连接 SQL Server 2005 的 JDBC 驱动 程序 是 ( D. 
A. com. mysql. jdbc. Driver 
B. sun. jdbc. odbc. JdbcOdbcDriver 
C. oracle. jdbc. driver. OracleDriver 
D. com. microsoft. sglserver. jdbc. SQLServerDriver 
2. 参考 6.8.2 节 中 的 主要 内 容 ,使 用 MySQL 创建 一 个 数据 库 yourdatabase, 并 在 该 数 
据 库 中 创建 一 张 表 student, 然 后 编写 程序 操作 该 表 。 
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6.9 小 结 


JDBC 连接 数据 库 有 两 种 常用 方式 : 建立 JDBC-ODBC 桥接 器 和 加 载 纯 Java 驱动 
程序 。 

目前 ,有 多 种 类 型 数据 库 。JDBC 不 管 连接 哪 种 类 型 的 数据 库 , 连 接 方 式 都 基本 类 
似 。 需 要 重点 关注 的 是 在 连接 各 数据 库 时 驱动 程序 加 载 部 分 的 代码 和 连接 部 分 的 
代码 。 

数据 库 连 接 池 的 基本 思想 是 : 为 数据 库 连 接 建 立 一 个 “缓冲 池 ”。 预 先 在 “缓冲 池 ” 
中 放 入 一 定数 量 的 连接 , 当 需 要 建立 数据 库 连 接 时 ,只 需 从 “缓冲 池 ” 中 取出 一 个 ,使 
用 完毕 之 后 再 放 回 去 。 可 以 通过 设 定 连接 池 最 大 连接 数 来 防止 系统 无 限度 地 与 数 
据 库 连 接 。 更 为 重要 的 是 ,通过 连接 池 的 管理 机 制 监视 数据 库 连 接 的 数量 及 使 用 情 
况 , 为 系统 开发 ,测试 和 性 能 调整 提供 依据 。 

预 处 理 语句 对 象 不 仅 大 大 提高 了 数据 库 的 执行 速度 ,而 且 方 便 编写 程序 。 


JSP 5 Servlet 


Java Servlet 的 核心 思想 就 是 在 Web 服务 器 端 创建 用 来 响应 用 户 请 求 的 对 象 ,该 对 象 
被 称 为 一 个 Servlet 对 象 。JSP 技术 以 Java Servlet 为 基础 , 当 客 户 请 求 一 个 JSP 页 面 时 ， 
Web 服务 器 (如 Tomcat 服务 器 ) 会 自动 生成 一 个 对 应 的 Java 文件 ,编译 该 Java 文件 ,并 用 
编译 得 到 的 字 节 码 文件 在 服务 器 端 创建 一 个 Servlet 对 象 。 但 大 多 数 Web 应 用 需要 
Servlet 对 象 具有 特定 的 功能 ,这 时 就 需要 Web 开发 人 员 自 己 编写 创建 Servlet 对 象 的 类 。 
本 章 将 重点 介绍 如 何 编写 Servlet 类 与 如 何 使 用 Servlet 类 。 


7.1 编写 Servlet 


7.1.1 知识 要 点 


编写 一 个 Servlet 类 很 简单 ,只 要 继承 javax. servlet. http 包 中 的 HttpServlet 类 ,并 重 
写 响 应 HTTP 请 求 的 方法 即 可 。HttpServlet 类 实现 了 Servlet 接口 ,实现 了 响应 用 户 的 接 
口 方 法 。HttpServlet 类 的 一 个 子 类 习惯 上 称 为 一 个 Servlet 类 ,这 样 的 子 类 创建 的 对 象 习 
惯 上 称 为 servlet 对 象 。 


7.1.2 技能 操作 


编写 一 个 简单 的 Servlet 类 MyFirstServlet, 用 户 请 求 这 个 servlet 对 象 时 ,就 会 在 浏览 
器 中 看 到 “人 生 第 一 个 Servlet 类 ”这 样 的 响应 信息 。 
代码 模板 MyFirstServlet. java 如 下 : 


package servlet; 
import java.io. * ; 
import javax.servlet. * ; 
import javax. servlet. http. * ; 
public class MyFirstServlet extends HttpServlet { 

public void init(ServletConfig config) throws ServletException( 

super. init(config) ; 
) 
public void service( HttpServletRequest request, HttpServletResponse response) 
throws IOException( 
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// 设 置 响应 的 内 容 类 型 

response. setContentType( "text/html;charset= utf-8") ; 
// 取 得 输出 对 象 

PrintWriter out 一 response. getWriter() ; 

out. println(" html — body"); 

// 在 浏览 器 中 显示 : 人 生 第 一 个 Servlet 类 

out.println(" 人 生 第 一 个 Servlet 3$") ; 

out. println("</body> </html>"); 


) 


编写 Servlet 类 时 必须 有 包 名 。 也 就 是 说 ,必须 在 包 中 编写 Servlet 类 。 在 本 章 中 我 们 
新 建 一 个 Web 工程 ch07, 所 有 的 Servlet 类 放 在 src 下 servlet 包 中 。 

技能 操作 中 Servlet 类 的 源 文件 MyFirstServlet Š 9 cho7 
. java 保存 在 Eclipse 的 Web 工程 ch07 的 src F servlet © . settings 
包 里 面 。MyFirstServlet. java 源 文件 由 Eclipse H & @ build 


动 编译 生成 字 节 码 文件 MyFirstServlet. class. 保存 = = agit 

在 buildVclassesVservlet HE Hi, Servlet 类 的 源 文件 ky WyFirstServlet. class 
与 字 节 码 文 件 保存 目录 如 图 7-1 所 示 。 编 写 完 Ce 

Servlet 类 的 源 文件 ,并 编译 了 源 文件 ,这 时 是 不 是 加 MyFirstServlet. java 
就 可 以 运行 servlet 对 象 呢 ? 不 可 以 ,需要 部 署 由 - 仑 YebContent 

servlet 之 后 才 可 以 运行 servlet 对 象 。 图 7-1 保存 目录 


7.1.3 拓展 训练 


1. servlet 对 象 是 在 服务 器 端 被 创建 的 还 是 在 用 户 端 被 创建 的 ? 

2. 编写 一 个 简单 的 Servlet 类 YourFirstServlet。 用 户 请 求 这 个 servlet 对 象 时 ,就 会 在 
浏览 器 中 看 到 “人 生 第 一 个 Servlet 类 ”这 样 的 响应 信息 ,并 在 Web 工程 中 找到 该 Servlet 类 
对 应 的 字 节 码 文 件 。 


7.2 部 署 与 运行 Servlet 


7.2.1 知识 要 点 


要 想 让 Web 服务 器 用 Servlet 类 编译 后 的 字 节 码 文件 创建 servlet 对 象 的 话 , 必 须 为 
Web 服务 器 编写 一 个 部 署 文件 。 这 个 部 署 文 件 是 一 个 XML 文件 ,名 字 是 web. xml, 该 文件 
由 Web 服务 器 负责 管理 。 在 ch07\ WebContent\WEB-INF 目录 里 找到 web. xml 文件 ,可 
以 在 该 文件 里 部 署 自己 的 servlet。 
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7.2.2 技能 操作 


掌握 部 署 与 运行 servlet 的 方法 ,具体 任务 如 下 : 
* WE servlet, 
。 运行 servlet。 
1. 部 署 servlet 
为 了 在 web. xml 文件 里 部 署 7. 1 节 中 的 MyFirstServlet, 需 要 在 web. xml 文件 里 
找到 过 web-app 之 过 /web-app 之 标记 ,然后 在 过 web-app 之 过 /web-app 之 标记 中 添加 如 
下 内 容 : 
<servlet> 
<servlet-name>myServlet</servlet-name> 
—servlet-class" servlet. MyFirstServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
—servlet-name2 myServlet— /servlet-name> 
url-pattern> /firstServlet< /url-pattern> 
</servlet-mapping> 
2. 运行 servlet 
在 web. xml 文件 中 部 署 完 servlet 之 后 ,就 可 以 运行 servlet T. servlet 第 一 次 被 直接 
访问 (可 以 通过 JSP 页 面 间接 访问 ) 时 ,需要 把 它 发 布 到 Web 
服务 器 上 (选中 Servlet 类 的 源 文件 , 右 击 ,选择 Run As 一 
Run on Server 命令 ) 。 这 时 ,在 Eclipse 内 内 的 浏览 器 中 看 到 
如 图 7-2 所 示 的 画面 。 网 人 放生 生得 站 于 
把 servlet 发 布 到 Web 服务 器 上 之 后 ,也 可 以 在 IE 浏览 器 的 地 址 栏 中 输入 : 


http: //localhost:8080/ch07 /firstServlet 


来 请 求 运行 servlet。 

为 了 更 好 地 理解 本 例 ,将 涉及 到 的 相关 知识 说 明 如 下 : 

(1) web. xml 文件 中 的 具体 内 容 及 其 作用 。 

(D RERE <web-app>. 

XML 文件 中 必须 有 一 个 根 标记 ,web. xml HIRERE <web-app>. 

© 二 servlet 二 标记 及 其 子 标记 。 

web. xml 文件 中 可 以 有 若干 个 二 servlet 二 标记 ,该 标记 的 内 容 由 Web 服务 器 负责 处 
JE, < servlet 二 标记 中 有 两 个 子 标 记 : < servlet-name > HI < servlet-class >, 其 中 
<servlet-name> FRIC HI A AE Web 服务 器 创建 的 servlet 对 象 的 名 字 。web. xml 文件 
中 可 以 有 若干 个 过 servlet 二 标记 ,但 要 求 其 二 servletrname 二 子 标记 的 内 容 互 不 相同 。 


http://localhost:8080/ch07/firstServlet 


人 生 第 一 个 Serviet 类 
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二 servlet-class 记 子 标记 的 内 容 指 定 Web 服务 器 用 哪个 类 来 创建 servlet 对 象 ,如 果 servlet 
对 象 已 经 创建 ,那么 Web 服务 器 就 不 再 使 用 指定 的 类 创建 。 

(9 一 servletrmapping 过 标记 及 其 子 标记 。 

web. xml 文件 中 若 出 现 一 个 二 servlet 之 标记 就 会 对 应 地 出 现 一 个 二 servletrmapping 之 
标记 。< 所 servletrmapping 之 标记 中 也 有 两 个 子 标记 : <servlet-name> H< url-pattern> , 
其 中 一 servletrname 二 子 标记 的 内 容 是 Web 服务 器 创建 的 servlet 对 象 的 名 字 ( 该 名 字 必 须 
Fl servlet fiie f) FERIE < servlet-name- ff P3 AK); — url-pattern Tb ie HKH 
定 用 户 用 怎样 的 模式 请 求 servlet 对 象 。 例 如 , < url-pattern 之 子 标记 的 内 容 是 
/firstServlet, 用 户 要 请 求 服务 器 运行 servlet 对 象 myServlet 为 其 服务 ,那么 在 IE 浏览 器 的 
地 址 栏 中 输入 : 


http: //localhost:8080/ch07/firstServlet 


— Web 服务 的 web. xml 文件 负责 管理 该 Web 服务 的 servlet XJ 28. 4 Web 服务 需 
要 提供 更 多 的 servlet 对 象 时 ,只 要 在 web. xml X44 P S In — servlet > RI < servlet-mapping— 9i 
记 就 可 以 。 

(2) servlet 的 生命 周期 。 

一 个 servlet 对 象 的 生命 周期 主要 由 下 列 三 个 过 程 组 成 。 

CD 初始 化 servlet 对 象 。 

当 servlet 对 象 第 一 次 被 请 求 加 载 时 ,服务 器 会 创建 一 个 servlet 对 象 ,该 servlet 对 象 调 
用 init 方法 完成 必要 的 初始 化 工作 。 

@ service 方法 响应 请 求 。 

创建 的 servlet 对 象 再 调用 service 方法 响应 客户 的 请 求 。 

© servlet 对 象 死亡 。 

当 服 务 器 关闭 时 ,servlet 对 象 调用 destroy 方法 使 自己 消亡 。 

从 上 面 三 个 过 程 来 看 ,init 方法 只 能 被 调用 一 次 , 即 在 servlet 第 一 次 被 请 求 加 载 时 调用 
该 方法 。 当 客户 请 求 servlet 服务 时 ,服务 器 将 启动 一 个 新 的 线程 ,在 该 线程 中 ,servlet 对 象 
调用 service 方法 响应 客户 的 请 求 。 那 么 多 个 客户 请 求 servlet 服务 时 ,服务 器 会 怎么 办 呢 ? 
服务 器 会 为 每 个 客户 启动 一 个 新 的 线程 ,在 每 个 线程 中 ,servlet 对 象 调用 service 方法 响应 
客户 的 请 求 。 也 就 是 说 ,每 个 客户 请 求 都 会 导致 service 方法 被 调用 执行 ,分 别 运 行 在 不 同 
的 线程 中 。 

(3) Servlet 类 中 的 方法 。 

O init 方法 。 

init 方法 是 HttpServlet 类 中 的 方法 ,可 以 在 Servlet 类 中 被 重 写 。init 方法 的 声明 格式 
WTF: 


public void init(ServletConfig config) throws ServletException 
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servlet 对 象 第 一 次 被 请 求 加 载 时 ,服务 器 创建 一 个 servlet 对 象 ,该 对 象 调用 init 方法 
完成 必要 的 初始 化 工作 。 

@ service 方法 。 

service 方法 是 HttpServlet 类 中 的 方法 ,可 以 在 Servlet 类 中 被 重 写 service 方法 的 声 
明 格 式 如 下 : 


public void service( HttpServletRequest request, HttpServletResponse response)throws IOException 


当 servlet 对 象 成 功 创建 后 ,servlet 对 象 就 调用 service 方法 来 处 理 用 户 的 请 求 并 返回 
响应 。 与 init 方法 不 同 的 是 ,init 方法 只 被 调用 一 次 ,而 service 方法 可 能 被 多 次 调用 。 我 们 
知道 , 当 客 户 请 求 servlet 对 象 时 ,服务 器 将 启动 一 个 新 的 线程 ,在 该 线程 中 ,servlet 对 象 调 
用 service 方法 响应 客户 的 请 求 。 换 句 话 说 ,每 个 客户 的 每 次 请 求 都 导致 service 方法 被 调 
用 执行 ,调用 过 程 运行 在 不 同 的 线程 中 , 互 不 干扰 。 

(3) destroy 方法 。 

destroy 方法 是 HttpServlet 类 中 的 方法 。 一 个 Servlet 类 可 直接 继承 该 方法 ,一般 不 需 
要 重 写 。destroy 方法 的 声明 格式 如 下 : 


public void destroy() 


当 服务 器 终止 服务 时 ,destroy 方法 会 被 执行 ,使 servlet 对 象 消亡 。 
7.2.3 拓展 训练 


1. 简 述 servlet 的 生命 周期 与 运行 原理 。 

2. servlet 对 象 初始 化 时 是 调用 init 方法 还 是 service 方法 ? 

3. 编写 一 个 简单 的 Servlet 类 Pratice2Servlet ,在 web. xml 中 部 署 该 servlet 并 运行 它 。 
当 用 户 通 过 在 IE 浏览 器 地 址 栏 中 输入 http://localhost: 8080/ch07/pratice2 请 求 这 个 
servlet 对 象 时 ,就 会 在 浏览 器 中 看 到 “部 署 与 运行 servlet” 这 样 的 响应 信息 。 


7.3 通过 JSP 页 面 访问 servlet 


可 以 通过 JSP 页 面 的 表单 或 超 链接 请 求 某 个 servlet。 通 过 JSP 页 面 访问 servlet 的 好 
处 是 ,JSP 页 面 负责 页 面 的 静态 信息 处 理 , 动 态 信 息 处 理由 servlet 完成 。 本 章 所 涉及 的 JSP 
页 面 均 保存 在 ch07\WebContent 目录 里 。 


7.3.1 知识 要 点 


1. 通过 表单 访问 servlet 
假设 在 JSP 页 面 中 ,有 如 下 表单 : 
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<form action— "isLogin" method= "post" > 
</form> 
那么 该 表单 的 处 理 程序 (action) 就 是 一 个 servlet, 为 该 servlet 部 署 时 ,web. xml 文件 中 的 
Ty id < servlet-mapping fj FERIE — url-pattern- fff] H AR /isLogin" 
2, 通过 超 链 接 访 问 servlet 
在 JSP 页 面 中 ,可 以 单 击 超 链接 ,访问 servlet 对 象 ,同时 也 可 以 通过 超 链 接 向 servlet 
提交 信息 ,例如 ,二 a href= "isLogin? user— tangtang&- &-pwd= 123456" > # f H] P AME 


I< /a> "EH PCR "CPGE BERE SRE user 一 tangtang 和 pwd= 123456 两 个 信息 
提交 给 servlet 处 理 。 


7.3.2 技能 操作 


灵活 使 用 JSP 页 面 访问 servlet 对 象 ,具体 任务 如 下 : 

编写 一 个 JSP 页 面 login. jsp, 在 该 页 面 中 通过 表单 向 名 字 为 login 的 servlet 对 象 (由 
LoginServlet 类 负责 创建 ) 提 交 用 户 名 和 密码 ,servlet 负责 判断 输入 的 用 户 名 和 密码 是 否 正 
确 , 并 把 判断 结果 返回 给 客户 。 需 要 为 web. xml 文件 添加 如 下 的 子 标记 : 


<servlet> 
<servlet-name> login</servlet-name> 
servlet-class> servlet. LoginServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name> login</servlet-name> 
<url-pattern>/isLogin</url-pattern> 
</servlet-mapping> 


页 面 运 行 效果 如 图 7-3、 图 7-4 和 图 7-5 所 示 。 


http://localhost:8080/ch07/loginjsp 


用 户 名 ， MÀ — ———— 
SS 一 hitp//localhost8080/ch07fisLogin http://localhost:8080/ch07/isLogin 
[sm 信息 输入 错误 信息 输入 正确 

图 7-3 信息 输入 页 面 图 7-4 servlet 错误 信息 7-5 servlet 正确 信息 

代码 模板 login. jsp 如 下 : 

<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 

"GBK"% > 

<html> 


<head> 
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<title>login. jsp</title> 


</head> 
<body> 
<form action= "isLogin" method= "post" > 
<table> 
<tr> 
«ud HP A: </td> 
<td><input type="text" name= "user"/ ></td> 
</tr> 
< t> 
ud 4B: </td> 
<td><input type="password" name= "pwd"/7 — /td> 
</tr> 
<tr> 
<td><input type="submit" value— "1 22" /7 — /td> 
<td><input type— "reset" value— "3E E" / 7 — / td 
</tr> 
</table> 
</form> 
</body> 
</html> 


代码 模板 LoginServlet. java 如 下 : 


package servlet; 
import java.io. * ; 
import javax.servlet. * ; 
import javax. servlet. http. * ; 
public class LoginServlet extends HttpServlet ( 
public void init(ServletConfig config) throws ServletException( 
super. init(config) ; 
) 
public void service( HttpServletRequest request, HttpServletResponse response) 
throws IOException( 
response. setContentType( "text/html ;charset— GBK") ; 
PrintWriter out= response. getWriter() ; 
request. setCharacterEncoding( " GBK") ; // 设 置 编码 ,防止 中 文 乱码 
String name= request. getParameter("user"); // 获 取 客 户 提交 的 信息 
String password= request. getParameter("pwd") ; // 获 取 客 户 提交 的 信息 
out. println(" html — body ") ; 
if(name— — null| | name. length) = —0)( 
out. println(" 请 输入 用 户 名 "); 
) 
else if( password — null| | password. length) = —0)( 


out. println(" 请 输入 密码 ") ; 
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else if( name. length) 7708. & password. length( ) 70) ( 
if(name. equals( " tangtang" ) & b. password. equals("123456") )( 
out. println(" 信 息 输入 正确 "); 
)else( 
out. println(" 信 息 输 入 错误 "); 
) 
) 
out. println( "</body> </html>"); 


) 


需要 特别 注意 的 是 ,如 果 servlet 的 请 求 格 式 是 /XXX”( 请 求 格式 就 是 web. xml 文件 
中 的 标记 过 servletrmapping 二 的 子 标记 二 url-pattern 二 的 内 容 ) ,那么 JSP 页 面 请 求 servlet 
时 ,必须 写成 "XXX”, 不 可 以 写成 "/XXX”, 和 否则 将 变 成 请 求 服务 器 (Tomcat)root 目录 下 的 


某 个 servlet, 
7.3.3 拓展 训练 
把 技能 操作 任务 中 login. jsp 的 信息 提交 方式 (表单 提交 ) 改 成 超 链接 提交 方式 来 访问 


servlet, 


提示 : 超 链 接 访 问 servlet 格式 如 下 : 


<a href— "isLogin?user— tangtang&-G.pwd — 123456" > T Hl P 4 RU i — /a 


7.4 doGet 和 doPost 方法 


在 编写 Servlet 类 时 ,经 常会 重 写 HttpServlet 类 中 的 doGet 和 doPost 方法 ,用 来 处 理 
客户 的 请 求 并 作出 响应 。 


7.4.1 知识 要 点 


当 服 务 器 接收 到 一 个 servlet 请 求 时 ,就 会 产生 一 个 新 线程 ,在 这 个 线程 中 让 servlet 对 
象 调用 service 方法 为 请 求 作 出 响应 。service 方法 首先 检查 HTTP 请 求 类 型 (get 或 post), 
并 在 service 方法 中 根据 用 户 的 请 求 方式 ,相应 地 再 调用 doGet 或 doPost 方法 。 

HTTP 请 求 类 型 为 get 方式 时 ,service 方法 调用 doGet 方法 响应 用 户 请 求 , HTTP 请 
求 类 型 为 post 方式 时 ,service 方法 调用 doPost 方法 响应 用 户 请 求 。 因 此 ,在 Servlet 类 中 ， 
没有 必要 重 写 service 方法 ,直接 继承 即 可 。 

在 Servlet 类 中 重 写 doGet 或 doPost 方法 来 响应 用 户 的 请 求 ,这 样 可 以 增加 响应 的 灵 
活性 ,减轻 服务 器 的 负担 。 
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7.4.2 技能 操作 


掌握 doGet 和 doPost 方法 的 调用 原理 ,具体 任务 如 下 : 

编写 一 个 JSP 页 面 input. jsp ,在 该 页 面 中 使 用 表单 向 servlet 对 象 computer 提交 和 矩形 
的 长 与 宽 的 值 。computer( 由 GetLengthOrAreaServlet 负责 创建 ) 处 理 手段 依赖 表单 提交 
数据 的 方式 , 当 提交 方式 为 get 时 ,computer 对 象 计算 矩形 的 周 长 ; 当 提交 方式 为 post 时 ， 
computer 对 象 计算 矩 形 的 面积 。 需 要 为 web. xml 文件 添加 如 下 的 子 标记 : 


<servlet> 

—servlet-name2* computer /servlet-name> 

—servlet-class" servlet. GetLengthOrA reaServlet — /servlet-class> 
</servlet> 
<servlet-mapping> 

“<servlet-name—> computer < /servlet-name> 

«url pattern» /showLengthOrArea< /url-pattern> 
</servlet-mapping> 


页 面 运行 效果 如 图 7-6 至 图 7-8 所 示 。 


http://localhost:8080/ch07/inputjsp — 

输入 矩形 的 长 和 宽 ， 提 交 给 servlet (post 方 式 ) 求 面积 : 
长 : 8 

# 5 

Ea 

输入 和 矩形 的 长 和 宽 ， 提 交 给 servlet (get 方 式 ) RAK: 
K 8 

宽 ; 5 

[s>] 

7-6 信息 输入 页 面 
http://localhost:8080/ch07/showLengthOrArea http;//localhost:8080/ch07/showLengthOrArea?length- 8&width- 5 
矩形 的 面积 是 :40. 0 矩形 的 周 长 是 ，26.0 

图 7-7 post 方式 提交 获得 矩形 面积 图 7-8 get 方式 提交 获得 矩形 周 长 


代码 模板 input. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 


5873€ JSP 5 Servlet 159 


<html> 
<head> 
<title>input. jsp</title> 
</head> 
<body> 
—h22 fi AKE BJ KIH, EXA servlet post 77 3X DOR Wi B1 : < /h22> 
<form action= "showLengthOrArea" method= "post" > 
K: «input type="text" name= "length" /7— br/7* 
宽 : <input type="text" name= "width" / 7 — br/7- 
<input type="submit" value— "122 "/7 
</form> 
<br/> 
<h2>—# A EE WIK, EXA servlet(get HORAK :</h2> 
<form action= "showLengthOrArea" method= "get" > 
长 : <input type="text" name= "length"/><br/> 
宽 : <input type="text" name= "width"/><br/> 
<input type— "submit" value=" #3 "/> 
</form> 
</body> 
</html> 


代码 模板 GetLengthOrAreaServlet. java 如 下 : 


package servlet; 
import java.io. * ; 
import javax. servlet. * ; 
import javax. servlet. http. * ; 
public class GetLengthOrAreaServlet extends HttpServlet { 
public void init(ServletConfig config) throws ServletException( 
super. init(config) ; 
) 
public void doPost( HttpServletRequest request, HttpServletResponse 
response) throws ServletException, IOException( 
response. setContentType( "text/html ;charset — GBK") ; 
PrintWriter out= response. getWriter() ; 
String l= request. getParameter( " length") ; 
String w= request. getParameter(" width") ; 
out. println(" html — body") ; 
double m=0, n=0; 
try{ 
m- Double. parseDouble(l) ; 
n= Double. parseDouble( w) ; 
out. printIn "Ë JE HRE: "十 mx n); // 计 算 矩 形 的 面积 ,并 在 浏览 器 上 输出 
) catch( NumberFormatException e) ( 


out. println(" 请 输入 数字 字符 !"); 
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out. println("  /body </html>"); 
) 
public void doGet( HttpServletRequest request, HttpServletResponse 
response) throws ServletException, IOExceptioní 
response. setContentType( " text/html ; charset— utf-8") ; 
PrintWriter out— response. getWriter() ; 
String l= request. getParameter( " length") ; 
String w= request. getParameter(" width") ; 
out. println("<html> <body>"); 
double m—0,n—0; 
tryl 
m= Double. parseDouble(l) ; 
n= Double. parseDouble( w) ; 
out. println C" JE HAKE: "十 (m 十 n) * 2); // 计 算 和 矩形 的 周 长 ,并 在 浏览 器 上 输出 
} catch( NumberFormatException e) ( 
out. println(" 请 输入 数字 字符 !"); 
) 
out. println( "</body> </html>"); 


) 

一 般 情况 下 ,不 论 用 户 请 求 类 型 是 get 还 是 post, 服 务 器 的 处 理 过 程 完 全 相同 ,那么 可 
以 只 在 doPost 方法 中 编写 处 理 过 程 ,而 在 doGet 方法 中 再 调用 doPost 方法 ; 或 只 在 doGet 
方法 中 编写 处 理 过 程 ,而 在 doPost 方法 中 再 调用 doGet 方法 。 


7.4.3 拓展 训练 


编写 一 个 JSP 页 面 pratice7_4. jsp, 在 该 JSP 页 面 中 用 户 可 以 使 用 表单 向 servlet Xf 4 
pratice 提交 矩形 的 长 与 宽 的 值 。pratice( 由 PraticeServlet 类 负责 创建 ) 处 理 数据 的 手段 不 
依赖 表单 提交 数据 的 方式 , 即 不 论 post 还 是 get, 处 理 数据 的 手段 相同 ,都 是 计算 矩形 的 
周 长 。 


7.5 重 定向 与 转发 


重 定向 是 将 用 户 从 当前 JSP 页 面 或 servlet 定向 到 另 一 个 JSP 页 面 或 servlet, 以 前 的 
request 中 存放 的 信息 全 部 失效 ,并 进入 一 个 新 的 request 作用 域 ; 而 转发 则 是 将 用 户 对 当 
前 JSP 页 面 或 servlet 的 请 求 转发 给 另 一 个 JSP 页 面 或 servlet, 以 前 的 request 中 存放 的 信 
息 不 会 失效 。 


7.5.1 知识 要 点 


1. 重 定向 
重 定向 方法 public void sendRedirect(String location) 是 HttpServletResponse 类 中 的 
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方法 。 重 定向 的 目标 页 面 或 servlet( 由 参数 location 指定 ) 无 法 从 以 前 的 request 对 象 获 取 
用 户 提交 的 数据 。 

2. 转发 

javax. servlet. RequestDispatcher 对 象 可 以 把 用 户 对 当前 JSP 页 面 或 servlet 的 请 求 转 
发 给 另 一 个 JSP 页 面 或 servlet。 实 现 转发 需要 两 个 步 又: 

。 获得 RequestDispatcher 对 象 。 

在 当前 JSP 页 面 或 servlet 中 ,使 用 request 对 象 调用 


public RequestDispatcher getRequestDispatcher(String url) 


方法 返回 一 个 RequestDispatcher 对 象 ,其 中 参数 url 就 是 要 转发 的 JSP 页 面 或 servlet 的 地 
址 ,例如 : 


RequestDispatcher dis= request. getRequestDispatcher( " dologin") ; 


* RequestDispatcher 对 象 调用 forward 方法 实现 转发 。 
获得 RequestDispatcher 对 象 之 后 ,就 可 以 使 用 该 对 象 调用 


public void forward(ServletRequest request, ServletResponse response) 


方法 将 用 户 对 当前 JSP 页 面 或 servlet 的 请 求 转发 给 RequestDispatcher 对 象 所 指定 的 JSP 
页 面 或 servlet。 例 如 : 


dis. forward(request, response) ; 


将 用 户 对 当前 JSP 页 面 或 servlet 的 请 求 转变 成 对 dologinCservleO 的 请 求 。 
7.5.2. 技能 操作 


理解 重 定 向 与 转发 的 区 别 , 掌 握 重 定向 与 转发 的 实现 方法 ,具体 任务 如 下 : 

编写 JSP 页 面 redirectForward. jsp. 在 该 JSP 页 面 中 通过 表单 向 名 为 rforward 的 
servlet 对 象 (由 RedirectForwardServlet 类 负责 创建 ) 提 交 用 户 名 和 密码 。 如 果 用 户 输入 的 
数据 不 完整 (没有 输入 用 户 名 或 密码 ) ,那么 rforward 就 将 用 户 重 定向 到 redirectForward 
.jsp 页 面 ; 如 果 用 户 输入 的 数据 完整 ,rforward 就 将 用 户 对 redirectForward. jsp 页 面 的 请 
求 转发 给 名 字 为 show 的 servlet 对 象 (由 ShowServlet 类 负责 创建 ) ,该 servlet 对 象 显示 用 
户 输入 的 信息 。 需 要 为 web. xml 文件 添加 如 下 的 子 标记 : 


servlet> 

<servlet-name> rforward< /servlet-name> 

servlet-class> servlet. RedirectForwardServlet— /servlet-class> 
</servlet> 
< 二 servlet 一 

—servlet-name2* show /servlet-name> 


162 网 络 编程 与 动态 网 站 制作 


“<servlet-class> servlet. ShowServlet< /servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name>rforward</servlet-name> 
<url-pattern>/rfLogin</url-pattern> 
</servlet-mapping> 
<servlet-mapping> 
<servlet-name>show</servlet-name> 
<url-pattern>/yourMessage</url-pattern> 
</servlet-mapping> 


代码 模板 redirectForward. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GB] 
"GBK" 42 


<html> 
<head> 
<title> redirectForward. jsp</ title 
</head> 
<body> 
<form action= "rfLogin" method= "post" > 
<table> 
<tr> 
<td> HPH: </td> 
td — input type="text" name= "user"/></td> 
</tr> 
<tr> 
去 td> 密 码 : </td> 
<td> — input type="password" name= "pwd"/></td> 
</tr> 
<tr> 
<td><input type="submit" value= "#3 "/></td> 
<td><input type= "reset" value— "3E E "/></td> 
</tr> 
</table> 
</form> 
</body> 
</html> 


代码 模板 RedirectForwardServlet. java 如 下 : 


package servlet; 

import java. io. IOException; 

import javax. servlet. RequestDispatcher; 
import javax. servlet. ServletConfig ; 


pageEncoding — 
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import javax. servlet. ServletException; 
import javax. servlet. http. HttpServlet; 
import javax. servlet. http. HttpServletRequest; 
import javax. servlet. http. HttpServletResponse; 
public class RedirectForwardServlet extends HttpServlet ( 
public void init(ServletConfig config) throws ServletException( 
super. init(config) ; 
) 
public void doPost( HttpServletRequest request, HttpServletResponse 
response) throws ServletException, IOException( 
String name= request. getParameter( "user" ) ; 
String password request. getParameter( " pwd") ; 
if(name— — null | name. length) = —0)( 
response. sendRedirect(" redirectForward. jsp") ; 
/ / fi Fl response 调用 sendRedirect 方法 重 定向 到 redirectForward. jsp 
) 
else if( password — null| | password. length) = =0) { 
response. sendRedirect(" redirectForward. jsp") ; 
// 使 用 response 调用 sendRedirect 方法 重 定向 到 redirectForward. jsp 
) 
else if(name. length() >0&.&.password.length()2>0) ( 
RequestDispatcher dis= request. getRequestDispatcher( " yourMessage") ; 
// 获 得 RequestDispatcher 对 象 dis, 转 发 到 servlet 对 象 yourMessage 
dis. forward( request, response) ; 


//dis 对 象 调用 forward 方法 实现 转发 


) 
public void doGet( HttpServletRequest request, HttpServletResponse 
response) throws ServletException, IOException( 
doPost( request, response) ; 


) 
代码 模板 ShowServlet. java 如 下 : 


package servlet; 
import java.io. * ; 
import javax.servlet. * ; 
import javax. servlet. http. * ; 
public class ShowServlet extends HttpServlet { 
public void init(ServletConfig config) throws ServletException( 
super. init(config) ; 
) 
public void doPost( HttpServletRequest request, HttpServletResponse 
response) throws ServletException, IOException( 
response. setContentType( "text/html ; charset— GBK") ; 
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PrintWriter out= response. getWriter() ; 
String name 一 request. getParameter("user"); 
String password 一 request. getParameter(" pwd") ; 
byte b[ ] — name. getBytes("ISO-8859-1") ; 
name- new String(b, "UTF-8") ; 
byte bl[ ] = password. getBytes("ISO-8859-1") ; 
password new String(bl, "UTF-8"); 
out. println(" 您 的 用 户 名 是 : " -name) ; 
out. printIn(" br # (fj 8:83 J& : "A password) ; 
) 
public void doGet( HttpServletRequest request, HttpServletResponse 
response) throws ServletException, IOException( 
doPost( request, response) ; 


) 

和 重 定向 方法 不 同 的 是 ,使 用 转发 时 ,用 户 在 浏览 器 的 地 址 栏 中 不 能 看 到 forward 方法 
所 转发 的 页 面 地 址 或 servlet 的 地 址 ,只 能 看 到 它们 的 运行 效果 。 用 户 在 浏览 器 的 地 址 栏 中 
所 看 到 的 仍然 是 当前 JSP 页 面 或 servlet 的 地 址 。 
7.5.3. de ASA 

1. 什么 是 转发 ? 什么 是 重 定向 ? 它们 有 什么 区 别 ? 

2. 试 着 把 技能 操作 任务 中 的 转发 改 成 重 定向 ,然后 运行 redirectForward. jsp 页 面 , 看 
看 运行 结果 是 什么 样 的 ,为 什么 是 这 样 的 结果 ? 


7.6 session 会 话 管理 


7.6.1 知识 要 点 

在 servlet 中 可 以 使 用 request 对 象 来 调用 getSession 方法 获取 用 户 的 会 话 对 象 
session ,例如 : 

HttpSession session= request. getSession(true) ; 

一 个 用 户 在 同一 个 Web 服务 中 的 不 同 servlet 对 象 中 获取 的 session 对 象 是 完全 相同 
的 ,但 是 不 同 用 户 的 session 对 象 互 不 相同 。 
7.6.2 技能 操作 

掌握 如 何在 servlet 中 获取 用 户 的 会 话 对 象 session, 具 体 任务 如 下 : 

编写 一 个 JSP 页 面 useSession. jsp. 在 该 页 面 中 通过 表单 向 名 字 为 useSession 的 
servlet 对 象 ( 由 UseSessionServlet 类 负责 创建 ) 提 交 用 户 名 , useSession 将 用 户 名 存 入 用 户 
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的 session 对 象 中 ,然后 用 户 请 求 男 一 个 servlet 对 象 showName( 由 ShowNameServlet 类 负 
责 创建 ),showName 从 用 户 的 session 对 象 中 取出 存储 的 用 户 名 ,并 显示 在 浏览 器 中 。 需 要 
为 web. xml 文件 添加 如 下 的 子 标记 : 


<servlet2> 
<serylet-name>useSession</servlet-name> 
<servylet-class> servlet. UseSessionServlet</servlet-class> 

</servlet> 

<servlet2> 
servlet-name> showName- /servlet-name> 
servlet-class> servlet. ShowNameServlet< /servlet-class> 

</servlet> 

<servlet-mapping> 
<servlet-name>useSession</servlet-name> 
<url-pattern>/sendMyName</url-pattern> 

</servlet-mapping> 

<servlet-mapping> 
<servlet-name>showName< /servlet-name> 
<url-pattern>/showMyName< / url-pattern 

< /servlet-mapping— 


http://localhost:8080/ch07/useSessionjsp 


APA: Suc 
GER] 


程序 运行 效果 如 图 7-9 至 图 7-11 所 示 。 图 7-9 信息 输入 页 面 
http://localhost:8080/ch07/sendMyName 
| http://localhost:8080/ch07/showMyName 
您 请 求 的 servlet 对 象 是 ，useSession 
会 话 ID 是 ，F97E2DB26869F34B86D502BDABTFD41D 您 请 求 的 servlet 对 象 是 ，showName 
请 音 请 求 另 一 个 servlet， 您 的 会 话 ID 是 ，F97E2DB26869F34B86D502BDAB7FD41D 
请 求 另 一 个 servlet 您 的 会 话 中 存储 的 用 户 名 是 : 雪山 飞狐 
图 7-10 获取 会 话 并 存储 数据 图 7-11 获取 会 话 中 的 数据 并 显示 


代码 模板 useSession. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title> useSession. jsp</title> 
</head> 
<body> 
<form action= "sendMyName" method="post" > 
<table> 
xu 
<td> HPA: </td> 
<td><input type="text" name= "user"/></td> 
</tr> 
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<tr> 
<td><input type="submit" value=" Z "/></td> 
</tr> 
</table> 
</form> 


</body> 
</html> 


代码 模板 UseSessionServlet. java 如 下 : 


package servlet; 


import java.io. * ; 


import javax.servlet. * ; 


import javax. servlet.http. * ; 
public class UseSessionServlet extends HttpServlet ( 
public void init(ServletConfig config) throws ServletException( 


) 


super. init(config) ; 


public void doPost( HttpServletRequest request, HttpServletResponse 


) 


response) throws ServletException, IOException( 
response. setContentType( " text/html ; charset — GBK"); 
PrintWriter out— response. getWriter() ; 
request. setCharacterEncoding( " GBK") ; 
String name= request. getParameter( " user") ; 
HttpSession session— request.getSession(true) ; ”// 获 得 用 户 的 会 话 对 象 
session. setAttribute("myName", name); 
out. println( " htm — body ") ; 
out.println(" 您 请 求 的 servlet X1 # Jë : "十 getServletName()); 
out. println(" 一 br 之 您 的 会 话 ID Æ: "十 session.getId()); 
out. printlIn( "<br if 3d iioK 5) — P servlet: ") ; 
out. println( "<br><a href— showMyName- ÑR 5j — A“ servlet</a>"); 
out. printIn( "  /body2 — / htm"); 


public void doGet( HttpServletRequest request, HttpServletResponse 


) 


response) throws ServletException, IOException( 
doPost( request, response) ; 


代码 模板 ShowNameServlet. java 如 下 : 


package servlet; 


import java.io. * ; 


import javax.servlet. * ; 


import javax.servlet. http. * ; 


public class ShowNameServlet extends HttpServlet ( 
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public void init(ServletConfig config) throws ServletException{ 
super. init(config) ; 
) 
public void doPost( HttpServletRequest request, HttpServletResponse 
response) throws ServletException, IOException( 
response. setContentType( "text/html ;charset— GBK") ; 
PrintWriter out= response. getWriter() ; 
HttpSession session— request. getSession(true) ; ”// 获 得 用 户 的 会 话 对 象 
String name= (String) session. getAttribute("myName"); 
out. println("<htm> <body>"); 
out. println(" 您 请 求 的 servlet 对 象 是 : "十 getServletName()); 
out. printIn(" br fi fj eii ID Æ: "十 session. getldO); 
out. println( "<br> f& ff] e i P T£ SE 09 AE: "A name); 
out. printlIn(" /body- — /htm2") ; 
) 
public void doGet( HttpServletRequest request, HttpServletResponse 
response) throws ServletException, IOException( 
doPost( request, response) ; 


) 


用 户 的 会 话 对 象 session 可 以 在 JSP 页 面 中 不 用 声明 直接 使 用 ,而 在 Servlet 类 中 必须 
先 使 用 request 对 象 获得 用 户 的 会 话 对 象 ,然后 再 使 用 它 。 


7.6.3 拓展 训练 


l. servlet 对 象 如 何 获得 用 户 的 会 话 对 象 ? 

2. WERE JSP 页 面 中 使 用 会 话 对 象 session 和 在 servlet 中 使 用 会 话 对 象 session 有 
什么 不 同 ,并 举例 说 明 。 

3. 假设 创建 servlet 的 类 是 my. servlet. MyFirstServlet ,创建 的 servlet 对 象 的 名 字 是 
first, 如 果 使 用 表单 请 求 该 servlet ,表单 的 action 的 值 是 isgo。 应 该 怎么 为 该 servlet 编写 
部 署 文 件 web. xml? 


7.7 小 结 


。 Java Servlet 技术 的 核心 思想 就 是 在 服务 器 端 创建 能 响应 用 户 请 求 的 对 象 ,该 对 象 
习惯 上 被 称 为 一 个 servlet 对 象 。 

。 要 想 让 Web 服务 器 用 Servlet 类 编译 后 的 字 节 码 文件 创建 servlet 对 象 , 必须 为 
Web 服务 器 编写 一 个 部 署 文件 web. xml, 

。 servlet 对 象 第 一 次 被 请 求 加 载 时 .服务 器 创建 一 个 servlet 对 象 , 该 对 象 调用 init 方 
法 完成 必要 的 初始 化 工作 。 当 servlet 对 象 成 功 创建 后 ,servlet 对 象 就 调用 service 
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方法 来 处 理 用 户 的 请 求 并 返回 响应 。 

。 当 服 务 器 接收 到 一 个 servlet 请 求 时 ,就 会 产生 一 个 新 线程 ,在 这 个 线程 中 让 servlet 
对 象 调用 service 方法 为 请 求 作出 响应 。 实 际 上 ,service 方法 首先 检查 HTTP 请 求 
类 型 (get、post 等 ) ,并 在 service 方法 中 根据 用 户 的 请 求 方式 ,对 应 地 再 调用 doGet 
或 doPost 方法 。 

。 重 定向 的 功能 是 将 用 户 从 当前 JSP 页 面 或 servlet 定向 到 另 一 个 JSP 页 面 或 
servlet, 以 前 的 request 中 存放 的 变量 全 部 失效 ,并 进入 一 个 新 的 request 作用 域 ; 
转发 的 功能 是 将 用 户 对 当前 JSP 页 面 或 servlet 的 请 求 转发 给 另 一 个 JSP 页 面 或 
servlet, 以 前 的 request 中 存放 的 变量 不 会 失效 。 

。 一 个 用 户 在 同一 个 Web 服务 中 的 不 同 servlet 中 获取 的 session 对 象 是 完全 相同 的 ， 
但 是 不 同 用 户 的 session 对 象 互 不 相同 。 


第 8 = 
基于 Servlet 的 MVC 模式 


前 面 的 章节 中 已 经 学 习 了 JSP 和 Servlet 技术 ,使 用 它们 可 以 开发 出 完整 的 Web 应 用 
程序 。 但 有 时 会 把 大 量 的 Java 代码 写 在 JSP 页 面 中 ,把 HTML 代码 写 在 Servlet 中 。 这 样 
会 造成 代码 编写 不 容易 ,日 后 维护 也 不 容易 。 因 此 ,学 习 Web 应 用 程序 的 设计 模式 就 显得 
尤为 重要 了 。 本 章 就 将 学 习 一 种 非常 典型 的 Web 应 用 程序 的 设计 模式 一 一 基于 Servlet 的 
MVC 模式 。 


8.1 JSP 中 的 MVC 模式 


8.1.1 知识 要 点 


1. MVC 的 概念 

MVC 是 Model, View „Controller 的 缩写 ,分 别 代表 Web 应 用 程序 中 的 三 种 职责 : 

。 模 型 一 一 用 于 存储 数据 以 及 处 理 用 户 请 求 的 业务 逻辑 。 

。 视图 一 一 向 控制 器 提交 数据 ,显示 模型 中 的 数据 。 

。 控制 器 一 一 根据 视图 提出 的 请 求 , 判 断 将 请 求 和 数据 交 给 哪个 模型 处 理 , 处 理 后 的 
有 关 结 果 交 给 哪个 视图 更 新 显示 。 

2. JSP 中 的 MVC 模式 

JSP 中 的 MVC 模式 的 具体 实现 如 下 : 

。 模型 。 一 个 或 多 个 JavaBean 对 象 , 用 于 存储 数据 (实体 模型 ,由 JavaBean 类 创建 ) 和 
处 理 业 务 逻 辑 ( 业 务 模 型 ,由 一 般 的 Java 类 创建 ) 。 

。 视图。 一 个 或 多 个 JSP 页 面 ,向 控制 器 提交 数据 和 为 模型 提供 数据 显示 ,JSP 页 面 
主要 使 用 HTML 标记 和 JavaBean 标记 来 显示 数据 。 

。 控制 器 。 一 个 或 多 个 servlet 对 象 ,根据 视图 提交 的 请 求 进行 控制 , 即 把 请 求 转发 给 
处 理 业务 逻辑 的 JavaBean ,并 将 处 理 结果 存放 到 实体 模型 JavaBean 中 ,输出 给 视图 
显示 。 

JSP 中 的 MVC 模式 的 流程 如 图 8-1 所 示 。 
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Web 服 务 器 
— | ”请求 | Servlet 
a E | -| (控制 器 ) 
| 
览 HTML 
" 响应 JSP m 
“j (视图 ) [ones 
š 显示 数据 | JavaBean 
(模型 ) 


8-1 JSP 中 的 MVC 模式 


8.1.2 技能 操作 


运用 MVC 模式 ,实现 登录 验证 过 程 。 具 体 任务 如 下 : 

编写 一 个 简单 的 Web 应 用 程序 : 用 户 登 录 验 证 程序 。 视 图 login. jsp 提交 数据 请 求 (用 
户 名 和 密码 ); 控制 器 controllerServlet(ControllerServlet 类 负责 创建 ) 接 收 请 求 信息 ,然后 
把 请 求 信息 封装 在 user(UserBean 类 负责 创建 ) 实 体 模型 中 ,控制 器 把 user 模型 传递 给 
userCheck 业务 模型 去 处 理 (UserCheckBean 类 负责 创建 ); 如 果 用 户 名 和 密码 输入 正确 , 则 
返回 success. jsp 页 面 ,否则 返回 login. jsp 页 面 。 程 序 运行 效果 如 图 8-2 所 示 。 


http://localhost:8080/ch08/loginjsp 
欢迎 登录 本 系统 ， 请 输入 用 户 名 与 密码 ! 
http://localhost:8080/ch08/isLogin 
用 户 名 ， 
恭喜 JSPIVC 登 录 成 功 ! 
LED 
zx [si] 
图 8-2 ”登录 验证 程序 
代码 模板 web. xml 如 下 : 
<?xml version— "1.0" encoding— "UTF-8"?> 
<web-app> 
<servlet> 


“<servlet-name>controllerServlet< /servlet-name> 
“<servlet-class>servlet. ControllerServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name> controllerServlet< /servlet-name> 
url-pattern> /isLogin</url-pattern> 
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< /servlet-mapping 
«/ web-app— 


代码 模板 UserBean. java( 实 体 模型 ) 如 下 : 


package entity. bean; 
public class UserBean ( 
String name; 
String pwd; 
public UserBean() ( 
) 
public String getName() | 
return name; 
) 
public void setName(String name) ( 
this. name — name; 
) 
public String getPwd() ( 
return pwd; 
) 
public void setPwd(String pwd) { 
this. pwd = pwd; 


) 
代码 模板 UserCheckBean. java( 业 务 模型 ) 如 下 : 


package busynees. bean; 
import entity. bean. UserBean; 
public class UserCheckBean ( 
// 验 证 登录 
public boolean validate( UserBean user) í 
ifCuser! = null&&-user. getName() . equals( "JSEMVC"))( 
if(user. getPwd() . equals(" MVC")) í 
return true; 
) 
return false; 
) 


return false; 


) 
代码 模板 login. jsp( 视 图 ) 如 下 : 
<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 


"GBK"% > 
<html> 
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<head> 
<title>login. jsp</title> 
</head> 
<body> 
<form action= "isLogin" method=" post" > 
欢迎 登录 本 系统 ,请 输入 用 户 名 与 密码 ! 
<table> 
<tr> 
ud HPZ: </td> 
<td><input type="text" name— "name" /77 </td> 
</tr> 
<> 
ud: </td> 
<td><input type="password" name= "pwd"/></td> 
</tr> 
<tr> 
<td><input type="submit" value= "f 22 "/></td> 
<td><input type= "reset" value=" Ņ H "/></td> 
</tr> 
</table> 
</form> 
</body> 
</html> 


代码 模板 success. jsp( 视 图 ) 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBI 
"GBK" 42 

<%@ page import— "entity. bean. UserBean" % > 

<html> 

<head> 

<title> success. jsp / title 

</head> 

<body> 

<jsp:useBean id= "userBean" type= "entity. bean. UserBean" scope= "request" /> 
恭喜 一 jsp:getProperty property= "name" name= "userBean" / 7 #4 3i RI) ! 
</body> 

</html> 


代码 模板 ControllerServlet. java( 控 制 器 ) 如 下 : 


pageEncoding = 


package servlet; 

import java. io. IOException; 

import java. io. Print Writer; 

import javax.servlet. RequestDispatcher; 
import javax. servlet. ServletConfig ; 
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import javax. servlet. ServletException; 
import javax. servlet. http. HttpServlet; 
import javax. servlet. http. HttpServletRequest; 
import javax. servlet. http. HttpServletResponse; 
import busynees. bean. UserCheckBean; 
import entity. bean. UserBean; 
public class ControllerServlet extends HttpServlet ( 
public void init(ServletConfig config) throws ServletException( 
super. init(config) ; 
) 
public void doPost( HttpServletRequest request, HttpServletResponse response) 
throws IOException, ServletException( 


request. setCharacterEncoding( " GBK") ; // 设 置 编码 ,防止 中 文 乱码 
String name= request. getParameter( "name" ) ; // 获 取 客 户 提交 的 信息 
String password= request. getParameter("pwd"); ”// 获 取 客 户 提交 的 信息 
UserBean user— new UserBean() ; // 实 例 化 实体 模型 user 
user. setName( name) ; // 把 数据 存在 模型 user 中 
user. setPwd(password) ; // 把 数据 存在 模型 user 中 


UserCheckBean userCheck=new UserCheckBean(); // 实 例 化 业务 模型 userCheck 
if(userCheck. validate(user)) { 
request.setAttribute("userBean", user); // 把 装 有 数据 的 实体 模型 user 
RequestDispatcher dis= request. getRequestDispatcher( " success. jsp") ; 
dis. forward( request, response) ; 
}else{ 
response. sendRedirect( "login. jsp") ; 


) 
public void doGet( HttpServletRequest request, HttpServletResponse response) 
throws IOException, ServletException( 
doPost( request, response) ; 


) 


TE JSP 的 MVC 模式 中 ,控制 器 servlet 创建 的 实体 模型 JavaBean 也 涉及 到 生命 周期 ， 
生命 周期 分 别 为 request, session 和 application。 下 面 以 技能 操作 任务 中 的 实体 模型 user 
来 讨论 这 三 种 生命 周期 的 模型 的 用 法 。 

* request 周期 的 模型 。 

使 用 request 周期 的 模型 一 般 需 要 以 下 几 个 环节 : 

(D 创建 模型 并 把 数据 保存 到 模型 中 。 

在 servlet 中 需要 这 样 的 代码 : 

UserBean user=new UserBean(); // 实 例 化 模型 user 


user. setName(name) ; // 把 数据 存在 模型 user 中 
user. setPwd( password) ; // 把 数据 存在 模型 user 中 
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@ 将 模型 保存 到 request 对 象 中 并 转发 给 视图 JSP. 
在 servlet 中 需要 这 样 的 代码 : 


request. setAttribute("userBean", user); // 把 装 有 数据 的 模型 user 输出 给 视图 success.jsp 页 面 

RequestDispatcher dis= request. getRequestDispatcher("success.jsp"); 

dis. forward(request, response) ; 

request, setAttribute(" userBean" ，user) 这 名 代码 指定 了 查找 JavaBean 的 关键 字 ,并 
决定 了 JavaBean 的 生命 周期 为 request。 

O 视图 更 新 。 

servlet 所 转发 的 页 面 ,例如 success. jsp 页 面 ,必须 使 用 useBean 标记 获得 servlet 所 创 
建 的 JavaBean 对 象 (视图 不 负责 创建 JavaBean), FE JSP 页 面 需要 使 用 这 样 的 代码 : 


<jsp:useBean id= "userBean" type= "entity. bean. UserBean" scope= "request" />> 

<jsp:getProperty property= "name" name= "userBean" /2> 

标记 中 的 id 就 是 servlet 所 创建 的 模型 JavaBean, 它 和 request 对 象 中 的 关键 字 对 应 。 
因为 在 视图 中 不 创建 JavaBean 对 象 , 所 以 在 useBean 标记 中 使 用 type 属性 ,而 不 使 用 class 
属性 。useBean 标记 中 的 scope 必须 和 存储 模型 时 的 范围 (request) 一 致 。 

。 session 周期 的 模型 。 

使 用 session 周期 的 模型 一 般 需 要 以 下 几 个 环节 : 

(D 创建 模型 并 把 数据 保存 到 模型 中 。 

在 servlet 中 需要 这 样 的 代码 : 


UserBean user=new UserBean(); // 实 例 化 模型 user 
user. setName(name) ; // 把 数据 存在 模型 user 中 
user. setPwd( password) ; // 把 数据 存在 模型 user 中 


@ 将 模型 保存 到 session 对 象 中 并 转发 给 视图 JSP。 
在 servlet 中 需要 这 样 的 代码 : 


Session.setAttribute("userBean", user); // 把 装 有 数据 的 模型 user 输出 给 视图 
success.jsp 页 面 

RequestDispatcher dis= request. getRequestDispatcher( " success. jsp") ; 

dis. forward(request, response); 

session. setAttribute(" userBean", user) 3X i] fC i348 E T Æ 4X JavaBean 的 关键 字 ,并 
决定 了 JavaBean 的 生命 周期 为 session, 

O 视图 更 新 。 

servlet 所 转发 的 页 面 ,例如 success. jsp 页 面 ,必须 使 用 useBean 标记 获得 servlet 所 创 
建 的 JavaBean 对 象 (视图 不 负责 创建 JavaBean) 。 在 JSP 页 面 需 要 使 用 这 样 的 代码 : 


<jsp:useBean id= "userBean" type= "entity. bean. UserBean" scope= "session" /7 
<jsp:getProperty property— "name" name= "userBean" />> 
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标记 中 的 id 就 是 servlet 所 创建 的 模型 JavaBean, 它 和 session 对 象 中 的 关键 字 对 应 。 
因为 在 视图 中 不 创建 JavaBean 对 象 , 所 以 在 useBean 标记 中 使 用 type 属性 ,而 不 使 用 class 
属性 。useBean 标记 中 的 scope 必须 和 存储 模型 时 的 范围 (session) 一 致 。 

注意 : 对 于 生命 周期 为 session 的 模型 ,servlet 不 仅 可 以 使 用 RequestDispatcher 对 象 
转发 给 JSP 页面, 也 可 以 使 用 response 的 重 定向 方法 (sendRedirect) 定 向 到 JSP 页面。 

。 application 周期 的 模型 。 

使 用 application 周期 的 模型 一 般 需要 以 下 几 个 环节 : 

(D 创建 模型 并 把 数据 保存 到 模型 中 。 

在 servlet 中 需要 这 样 的 代码 : 


UserBean user 一 new UserBean() ; // 实 例 化 模型 user 
user. setName(name) ; // 把 数据 存在 模型 user 中 
user. setPwd( password) ; // 把 数据 存在 模型 user 中 


@ 将 模型 保存 到 application 对 象 中 并 转发 给 视图 JSP。 
在 servlet 中 需要 这 样 的 代码 : 


application. setAttribute("userBean", user); // 把 装 有 数据 的 模型 user 输出 给 视图 success. jsp 页 面 

RequestDispatcher dis= request. getRequestDispatcher( "success.jsp"); 

dis. forward(request, response) ; 

application. setAttribute("userBean" ，user) 这 句 代码 指定 了 查找 JavaBean 的 关键 字 ， 
并 决定 了 JavaBean 的 生命 周期 为 application, 

@ 视图 更 新 。 

servlet 所 转发 的 页 面 ,例如 success. jsp 页 面 ,必须 使 用 useBean 标记 获得 servlet 所 创 
建 的 JavaBean 对 象 (视图 不 负责 创建 JavaBean)。 在 JSP 页 面 需要 使 用 这 样 的 代码 : 

<jsp:useBean id= "userBean" type= "entity. bean. UserBean" scope= "application" />> 

<jsp:getProperty property= "name" name= "userBean"/> 

标记 中 的 id 就 是 servlet 所 创建 的 模型 JavaBean, © A application 对 象 中 的 关键 字 对 
应 。 因 为 在 视图 中 不 创建 JavaBean 对 象 , 所 以 在 useBean 标记 中 使 用 type 属性 ,而 不 使 用 
class 属性 。useBean 标记 中 的 scope 必须 和 存储 模型 时 的 范围 (application) 一 致 。 

注意 : 对 于 生命 周期 为 application 的 模型 ,servlet 不 仅 可 以 使 用 RequestDispatcher 对 
象 转发 给 JSP 页 面 ,也 可 以 使 用 response 的 重 定向 方法 (sendRedirect) 定 向 到 JSP 页 面 。 


8.1.3 拓展 训练 


1. 以 下 不 属于 MVC 设计 模式 中 3 个 模块 的 是 ( M 
A. 模型 B. 表示 层 C. 视图 D. 控制 器 
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2. 在 MVC 模式 中 ,( ) 用 于 客户 端 应 用 程序 的 图 形 数据 表示 ,与 实际 数据 处 理 
无 关 。 


A. 模型 B. 视图 C. 控制 器 D. 数据 
3. 在 MVC 设计 模式 中 ,( ) 接 收 用 户 请 求 数 据 。 
A. HTML B. JSP C. Servlet D. 业务 类 


4. 将 8.1.2 节 中 的 实体 模型 user 的 生命 周期 改 为 session ,并 运行 程序 。 
8.2 使 用 MVC 模式 查询 数据 库 


8.2.1 知识 要 点 


使 用 MVC 模式 设计 Web 应 用 时 ,尽量 把 实体 模型 与 业务 模型 分 开 实现 ,以 方便 以 后 
维护 。 例 如 ,在 使 用 MVC 模式 查询 数据 库 这 个 Web 应 用 中 数据 的 封装 由 实体 模型 Goods 
完成 ,而 处 理 数 据 由 业务 模型 GoodsDao 完成 和 数据 库 的 操作 由 业务 模型 DataBaseBean 
完成 。 

8.2.2 技能 操作 


使 用 MVC 模式 设计 Web 应 用 。 具 体 任务 如 下 : 

设计 一 个 Web 应 用 ,有 两 个 JSP 页 面 (addGoods. jsp 和 showAllGoods. jsp), =^ 
JavaBean( 实 体 模型 Goods、 业 务 模型 GoodsDao 和 业务 模型 DataBaseBean) 以 及 一 个 
servlet(GoodsControllerServlet) 。 用 户 在 JSP 页 面 addGoods. jsp 中 输入 商品 的 信息 ,提交 
给 servlet, 该 servlet 负责 添加 商品 (调用 业务 模型 GoodsDao 的 addGoods 方法 ) 查询 商品 
(调用 业务 模型 GoodsDao 的 getAllGoods 方法 ) ,并 把 查询 结果 显示 在 showAllGoods. jsp 
页 面 中 。Web 应 用 中 使 用 了 6.2 节 曾 经 使 用 过 的 数据 表 goodsInfo。 

页 面 运行 效果 如 图 8-3 和 图 8-4 所 示 。 


http://localhost:8080/ch08/addServlet 


商品 编号 商品 名 称 商品 价格 商品 类 别 


i FE [12.0 [BAR 
hitpi//localhost:S080/ch08/addGoods jsp 2 Pk [2500.0 | 电器 
12 E [18.5 Ra 
商品 编号 是 主键 ， 由 程序 自动 产生 ! a GET aeo KE 
qu ——sir 5 LEX [1800.0 ”服装 
masm: [aeu 6 Ba feo zd 
座 品 价格 : ss o 0 0 1] 7 ER “|17.0 KF 
商品 类 型 : [ra x no ERI [299.99 ”| 电器 
e (s=) a ee s.s Ra 


图 8-3 商品 信息 输入 页 面 图 8-4 商品 信息 显示 页 面 
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要 想 达到 上 述 效果 ,需要 为 web. xml 文件 添加 如 下 子 标记 : 


<servlet> 
<servlet-name>addServlet< /servlet-name> 
servlet-class> servlet. GoodsControllerServlet— /servlet-class> 
</servlet> 
<servlet-mapping> 
<serylet-name>addServlet</servlet-name> 
url-pattern> /addServlet</url-pattern> 
</servlet-mapping> 


代码 模板 Goods. java( 实 体 模型 ) 如 下 : 


package entity. bean; 
public class Goods í 

int goodsld; 

String goodsName; 

double goodsPrice; 

String goodsType; 

public Goods() ( 

) 

public int getGoodsId() ( 
return goodsld; 

) 

public void setGoodsId(int goodsId) { 
this. goodsId = goodsld; 

) 

public String getGoodsName() ( 
return goodsName; 

) 

public void setGoodsName(String goodsName) | 
this. goodsName — goodsName; 

) 

public double getGoodsPrice() | 
return goodsPrice; 

) 

public void setGoodsPrice(double goodsPrice) ( 
this. goodsPrice — goodsPrice; 

) 

public String getGoodsType(O | 
return goodsType; 

) 

public void setGoodsType(String goodsType) í 
this. goodsType = goodsType; 
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代码 模板 GoodsDao. java( 业 务 模型 ) 如 下 : 


package busynees. bean; 
import java.sql. * ; 
import java. util. ArrayList; 
import entity. bean. Goods; 
public class GoodsDao ( 
// 获 得 新 添加 的 商品 编号 
public synchronized int getID() { 
Connection con— DataBaseBean. getCon() ; 
PreparedStatement ps— null; 
ResultSet rs— null; 
int id—0; 
try ( 
ps- con. prepareStatement( "select max(goodsld) from goodsInfo") ; 
rs— ps. executeQueryO ; 
if(rs. next) ( 
id—rs.getInt(1) t-1; 
) 
) catch (SQLException e) { 
// TODO Auto-generated catch block 
e. printStackTrace() ; 
) finally( 
DataBaseBean. close(rs) ; 
DataBaseBean. close( ps) ; 
DataBaseBean. close(con) ; 
) 
return id; 
) 
// 添 加 商品 
public boolean addGoods(Goods goods) ( 
Connection con= DataBaseBean. getCon() ; 
PreparedStatement ps— null; 
try | 
ps= con. prepareStatement( "insert into goodsInfo values(?, ?, ?, ?)") ; 
ps.setInt(1, goods. getGoodsId()) ; 
ps.setString(2, goods. getGoodsName()) ; 
ps.setDouble(3, goods. getGoodsPrice()) ; 
ps.setString(4, goods. getGoodsType()) : 
int i= ps. executeUpdate( ) ; 
ifci0) 
return true; 
) catch (SQLException e) { 
// TODO Auto-generated catch block 
e. printStackTrace() ; 
) finallyt 
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DataBaseBean. close(ps) ; 
DataBaseBean. close(con) ; 
) 
return false; 
) 
// 查 询 商品 
public ArrayList<Goods> getAllGoods() { 
Connection con= DataBaseBean. getCon() ; 
PreparedStatement ps- null; 
ResultSet rs— null; 
ArrayList<Goods> al— new ArrayList- Goods () ; 
try ( 
ps 一 con. prepareStatement("select * from goodsInfo") ; 
rs— ps. executeQuery() ; 
while(rs. next) ) 1 
Goods gd= new GoodsO ; 
gd. setGoodsld(rs. getInt(1)) ; 
gd. setGoodsName(rs. getString(2)) ; 
gd. setGoodsPrice(rs. getDouble(3)) ; 
gd. setGoodsType(rs. getString(4)) ; 
al.add(gd) ; 
) 
) catch (SQLException e) | 
// TODO Auto-generated catch block 
e. printStackTrace() ; 
) finally( 
DataBaseBean. close(rs) ; 
DataBaseBean. close( ps) ; 
DataBaseBean. close(con) ; 
) 


return al; 


) 
代码 模板 DataBaseBean. java( 数 据 库 业 务 模型 ) 如 下 : 


package busynees. bean; 
import java. sql. Connection; 
import java. sql. DriverManager; 
import java. sql. PreparedStatement; 
import java. sql. ResultSet; 
import java. sql. SQLException; 
public class DataBaseBean | 
public static Connection getCon() ( 
Connection con=null; 
try { 
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Class. forName( "oracle. jdbc. driver. OracleDriver") ; 


) catch (ClassNotFoundException e) ( 
e. printStackTrace() ; 

) 

try ( 


con = DriverManager. getConnection ( " jdbc: oracle: thin: @ 127. 0. 0. 1: 1521: orcl", 


"system" , "system") ; 
) catch (SQLException e) { 
e. printStackTrace() ; 
) 
return con; 
) 
public static void close(ResultSet rs) í 
if(rs! — nulb { 
try { 
rs.close() ; 
) catch (SQLException e) | 
e. printStackTrace() ; 


) 


) 
public static void close( PreparedStatement ps) | 
if(ps!= null) ( 
try { 
ps. close) ; 
} catch (SQLException e) { 
e. printStackTrace() ; 


) 
public static void close( Connection con) | 
if(con! — null) ( 
try { 
con. close() ; 
) catch (SQLException e) { 
e. printStackTrace() ; 


) 
代码 模板 addGoods. jsp( 视 图 1) 如 下 : 


<% @ page language java" contentType 
"GBK"%> 


<html> 


text/html; charset = GBK pageEncoding = 
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<head> 
<title>addGoods. jsp / title 
</head> 
<body> 
—h4> Hi ñ fij Jë = RE, H E PF Ë 3Jj AE !< /h4—> 
<form action= "addServlet" method= " post" > 
<table border— "1"> 
<tr> 
<td> Wi ih APR </td> 
<td><input type="text" name= "goodsName" / 7 </td> 
</tr> 


<tr> 

二 td 商品 价格 :二 /td 二 

<td><input type="text" name= "goodsPrice"/></td> 
</tr> 


<e> 
<td> ql na KH : </td> 
<td> 
<select name= "goodsType"> 
<option value=" H Hiià "— Hih 
<option value= "i AK" — Hà AK 
<option value= "A m ">R h 
<option value= "KR ">K 
<option value 一 "服装 "之 服装 
<option value 一 "文具 "二 文具 
<option value 一 "其 他 "二 其 他 
</select> 
</td> 
</tr> 


<tr> 
<td><input type— "submit" value=" M" ></td> 
<td><input type= "reset" value=" "></td> 
</tr> 
</table> 
</form> 
</body> 
</html> 


代码 模板 showAllGoods. jsp( 视 图 2) 如 下 : 
<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 


"GBK"%> 
<%@ page import= "java. util. ArrayList" 967» 
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<%@ page import= "entity. bean. Goods" % > 
<html> 
<head> 
—title showAllGoods. jsp / title 
</head> 
<body> 
<% 
ArrayList<Goods> al=(ArrayList<Goods> ) request. getAttribute( "goods") ; 
ifCal! — null& &al. size() 70) ( 
%> 
<table border=1> 
<tr> 
<th> Rih [th 
<th> B ñh FR [th 
二 th 二 商品 价格 二 /th 二 
二 th 商品 类 别 二 /th 
</tr> 
<% 
for(int i=0;i<al. size() ;i++){ 
Goods good=al. get(i) ; 
6n 
<tr> 
<td>< Yi — good. getGoodsld() % ></td> 
<td>< Yi =good. getGoodsName() % ></td> 
<td>< Yi =good. getGoodsPrice() % ></td> 
<td>< Yi = good. getGoodsType() % ></td> 
</tr> 
<% 
) 
An 
</table> 
<% 
}else{ 
out. print(" 还 没有 商品 "); 
} 
%> 
</body> 
</html> 


代码 模板 GoodsControllerServlet. java( 控 制 器 ) 如 下 : 


package servlet; 

import java. io. IOException; 

import java. util. ArrayList; 

import javax. servlet. RequestDispatcher; 
import javax. servlet. ServletConfig ; 


import javax. 
import javax. 
import javax. 
import javax. 
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servlet. ServletException ; 

servlet. http. HttpServlet; 

servlet. http. HttpServletRequest; 
servlet. http. HttpServletResponse; 


import busynees. bean. GoodsDao; 
import entity. bean. Goods; 
public class GoodsControllerServlet extends HttpServlet ( 
public void init(ServletConfig config) throws ServletException( 
super. init(config) ; 


) 


public void doPost( HttpServletRequest request, HttpServletResponse response) 


throws IOException, ServletException( 


request. setCharacterEncoding( " GBK") ; // 设 置 编码 ,防止 中 文 乱码 
String goodsName= request. getParameter("goodsName"); // 获 得 视图 提交 的 信息 
String goodsPrice= request. getParameter("goodsPrice") ; 

String goodsType- request. getParameter("goodsType") ; 


Goods goods— new Goods() ; // 创 建 实体 模型 goods 

goods. setGoodsId(gd. getID()) ; // 把 数据 存储 到 实体 模型 中 

goods. setGoodsName( goodsName) ; // 把 数据 存储 到 实体 模型 中 

goods. setGoodsPrice( Double. parseDouble(goodsPrice) ); ”// 把 数据 存储 到 实体 模型 中 
goods. setGoodsType( goodsType) ; // 把 数据 存储 到 实体 模型 中 
GoodsDao gd= new GoodsDao() ; // 创 建 业务 模型 gd 


让 (gd.addGoods(goods)){ // 业 务 模型 gd 调用 addGoods 方法 添加 商品 


ArrayList<Goods> al— gd.getAllGoods(); // 业 务 模型 gd 调用 getAllGoods 方法 查询 商品 
request. setAttribute( "goods", al); // 把 商品 数组 保存 到 request 里 面 
RequestDispatcher dis= request. getRequestDispatcher("showAllGoods. jsp") ; 

dis. forward(request, response); 


}else{ 


) 
) 


response. sendRedirect( " addGoods. jsp") ; 


public void doGet( HttpServletRequest request, HttpServletResponse response) 


throws IOException, ServletException( 


doPost( request, response) ; 


) 


fi JH MVC 模式 设计 Web 应 用 时 控制 器 尽量 不 处 理 数据 , 它 只 是 起 到 控制 转发 的 功 
能 ,而 数据 交 给 业务 模型 处 理 , 如 技能 操作 任务 中 的 控制 器 。 


8.2.3 拓展 训练 


1. 使 用 MVC 模式 设计 Web 应 用 有 什么 好 处 ? 
2. MVC 中 的 模型 是 由 servlet 负责 创建 ,还 是 由 JSP 页 面 负责 创建 ? 
3. 使 用 MVC 模式 设计 一 个 Web 应 用 ,用 户 通过 JSP Xt ifii inputNumber. jsp 输入 两 个 
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操作 数 , 并 选择 一 种 运算 符 , 单 击 “ 提 交 ” 按 钮 后 ,调用 HandleComputer. java 这 个 servlet, 
在 HandleComputer. java 中 获取 用 户 输入 的 数字 和 运算 符 并 将 这 些 内 容 放 入 
ComputerBean. java 这 个 JavaBean 中 ,在 showResult. jsp 中 调用 JavaBean 显示 计算 的 
结果 。 


8.3 小 结 


MVC 模式 将 “视图 "“ 模 型 ?和 "控制 器 ?有效 地 组 合 。 在 JSP 的 MVC 模式 中 ,视图 是 
一 个 或 多 个 JSP 页 面 ,作用 是 向 控制 器 提交 数据 和 为 模型 提供 数据 显示 ; 模型 是 一 个 或 多 
个 JavaBean 对 象 , 用 于 存储 数据 (实体 模型 ,由 JavaBean 类 创建 ) 和 处 理 业 务 逻 辑 ( 业 务 模 
型 ,由 一 般 的 Java 类 创建 ); 控制 器 是 一 个 或 多 个 servlet 对 象 ,根据 视图 提交 的 请 求 进行 控 
制 , 即 把 请 求 转发 给 处 理 业 务 逻 辑 的 JavaBean, 并 将 处 理 结果 存 放 到 实体 模型 JavaBean 中 ， 
输出 给 视图 显示 。 


开发 Web 应 用 过 滤器 


在 Web 开发 过 程 中 ,可 能 有 这 样 的 需求 : 某 些 页 面 只 希望 几 个 特定 的 用 户 浏览 。 对 于 
这 样 的 访问 权限 的 控制 ,该 如 何 实现 呢 ? 过 滤器 (Filter) 就 可 以 完成 上 述 需求 的 实现 。 过 滤 
器 作用 于 服务 器 (Servlet) 处 理 请 求 之 前 或 服务 器 (Servlet) 响应 请 求 之 前 。 也 就 是 说 , 它 既 
可 以 过 滤 浏 览 器 对 服务 器 的 请 求 ,也 可 以 过 滤 服 务 器 对 浏览 器 的 响应 ,如 图 9-1 所 示 。 


Web 服 务 器 
pars I anaq 
HTTP 请 求 过 滤 后 的 请 求 Servlet 
w. 一 | 处 理 请 求 
器 
应 [过 | 过 滤 前 的 响应 
| _HTTPm 应 = B 过 滤 前 的 响应 响应 请 求 
器 
图 9-1 过 滤器 


如 何 编写 过 滤器 类 ,又 如 何 使 用 过 滤器 类 ,关于 这 些 将 在 本 章 重点 介绍 。 
9.1 Filter 类 与 filter 对 象 


9.1.1 知识 要 点 

编写 一 个 过 滤器 类 很 简单 ,只 要 实现 javax. servlet 包 中 的 Filter 接口 。 实 现 Filter 接 
口 的 类 习惯 上 称 为 一 个 Filter 类 ,这 样 的 类 创建 的 对 象 又 称 为 filter 对 象 。 
9.1.2 技能 操作 

编写 一 个 类 实现 Filter 接口 ,该 类 的 主要 功能 是 创建 过 滤器 。 具 体 任务 如 下 : 新 建 一 
个 Web 工程 ch09 ,在 该 Web 工程 中 编写 一 个 简单 的 Filter 类 MyFirstFilter, Filter 类 实现 


如 下 功能 : 
不 管用 户 请 求 该 Web 工程 的 哪个 页 面 或 servlet, 都 会 在 浏览 器 中 先 出 现 “ 首 先 执 行 过 


186 网 络 编程 与 动态 网 站 制作 


滤器 ?这 样 的 响应 信息 。 
代码 模板 MyFirstFilter. java 如 下 : 


package filters; 
import java. io. IOException; 
import java. io. PrintWriter; 
import javax. servlet. Filter; 
import javax. servlet. FilterChain; 
import javax. servlet. FilterConfig ; 
import javax. servlet. ServletException ; 
import javax. servlet. ServletRequest ; 
import javax. servlet. ServletResponse; 
public class MyFirstFilter implements Filter ( / / Sz i Filter 接口 的 类 
public void destroy() ( 
) 
public void doFilter(ServletRequest request, 
ServletResponse response, 
FilterChain chain) throws IOException, ServletException ( 
// 设 置 响应 类 型 
response. setContentType( " text/html ; charset=GBK"); 
// 获 得 输出 对 象 out 
PrintWriter out= response. getWriter() ; 
// 在 浏览 器 中 输出 
out. print(" 首 先 执 行 过 滤器 二 br 二 "); 
// 执 行 下 一 个 过 滤器 
chain. doFilter( request, response); 
} 
public void init(FilterConfig fConfig) throws ServletException { 
) 
) 


可 以 从 任务 中 MyFirstFilter 类 的 源 代码 看 出 : Filter 接口 与 Servlet 接口 很 类 似 , 同 样 
都 有 init() 与 destroy() 方 法 ,还 有 一 个 doFilter() 方 法 类 似 于 Servlet 接口 的 service OH 
法 。 下 面 分 别 介 绍 这 三 个 方法 的 功能 : 

(1) public void init(FilterConfig Config) throws ServletException 

方法 的 功能 是 初始 化 过 滤器 对 象 。 方 法 中 的 参数 {Config 是 FilterConfig 的 对 象 ,该 对 
象 代表 web. xml 中 为 过 滤器 定义 的 对 象 。 如 果 在 web. xml 中 为 过 滤器 设置 了 初始 参数 ， 
则 可 以 通过 FilterConfig 的 getInitParameter( String paramName) 方 法 获得 初始 参数 值 。 

(2) public void doFilter (ServletRequest request. ServletResponse response, FilterChain 
chain) throws IOException. ServletException 

2 Web 服务 器 使 用 servlet 对 象 调 用 service() 方 法 处 理 请 求 前 ,发 现 应 用 了 某 个 过 滤 
器 时 ,Web 服务 器 就 会 自动 调用 该 过 滤器 的 doFilter() 方 法 。 在 doFilter() 方 法 中 有 这 样 一 
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chain. doFilter(request, response) ; 


如 果 执 行 了 该 语句 ,就 会 执行 下 一 个 过 滤器 (根据 一 filtermapping 二 在 web. xml 中 出 
现 的 先后 顺序 执行 过 滤器 ); 如 果 没 有 下 一 个 过 滤器 ,就 返回 请 求 目 标 程序 。 如 果 因 为 某 个 
原因 没有 执行 “chain. doFilter(request, response) ;”, 则 请 求 就 不 会 继续 交 给 以 后 的 过 滤器 
或 请 求 目标 程序 ,这 就 是 所 谓 的 拦截 请 求 。 

(3) public void destroy() 

4 Web 服务 器 终止 服务 时 ,destroy 方法 会 被 执行 ,使 filter 对 象 消亡 。 


编写 完 Filter 类 的 源 文 件 ,并 编译 了 源 文件 ,这 时 Web 服务 器 是 不 是 就 可 以 运行 filter 
对 象 呢 ? 不 可 以 ,需要 部 署 filter 之 后 ,Web 服务 器 才 可 以 运行 filter 对 象 。 


9.1.3 拓展 训练 


尝试 找 一 下 技能 操作 任务 中 的 Filter 类 编译 后 的 字 节 码 文 件 。 


9.2 filter 对 象 的 部 署 与 运行 
9.2.1 知识 要 点 


要 想 让 Web 服务 器 用 Filter 类 编译 后 的 字 节 码 文件 创建 filter 对 象 , 必 须 在 Web 工程 
的 web. xml 文件 里 部 署 自己 的 filter, 
9.2.2 技能 操作 
部 署 和 运行 过 滤器 。 具 体 任务 如 下 : 
。 部 署 filter。 
。 运行 filter。 
1. 部 署 filter 


为 了 在 web. xml 文件 里 部 署 9. 1. 2 节 中 的 MyFirstFilter, 需 要 在 web. xml 文件 里 找 
ll — web-app > < /web-app > fi id. #Ñ Jš E < web-app > < /web-app > # iu "P1 38 Jill # F 
内 容 : 
«filter 
—filter-name myFirstFilter— /filter-name> 
<filter-class> filters. MyFirstFilter— /filter-class> 
</filter> 
<filter-mapping> 


<filter-name>myFirstFilter</filter-name> 
<url-pattern>/ * </url-pattern> 
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< /filter-mapping> 


H Y EA KU PANE ERRI , UNG filter bie RET sie, < filter-mapping> PREK 
其 子 标记 相关 内 容 说 明 如 下 。 
D «filter rid & RT sit 
web. xml 文件 中 可 以 有 若干 个 一 filter 之 标记 ,该 标记 的 内 容 由 Web 服务 器 负责 处 理 。 
<filter 过 标记 中 有 两 个 子 标 记 : <filter-name> fl — filter-class> , H rh < filtercname> F 
标记 的 内 容 是 Web 服务 器 创建 的 filter 对 象 的 名 字 。web. xml 文件 中 可 以 有 若干 个 
«filter bid .fH SK CITH — filter-name7 FERA ZEE AH I]. <filter-class> FER 
记 的 内 容 指定 Web 服务 器 用 哪个 类 来 创建 filter 对 象 ,如 果 filter 对 象 已 经 创建 ,那么 Web 
服务 器 就 不 再 使 用 指定 的 类 创建 。 
如 果 在 过 滤器 初始 化 时 ,需要 读 取 一 些 参数 的 值 , 则 可 以 在 坪 filter 标记 中 使 用 过 init- 
param 祖 子 标记 设置 。 例 如 : 
«filter 
filter-name2 myFirstFilter— /filter-name> 
—filter-class filters. MyFirstFilter— /filter-class> 
<init-param> 
—param-name encoding — /param-name> 
<param-value>GBK</param-value> 


</init-param> 
</filter> 


那么 就 可 以 在 filter 的 init() 方 法 中 使 用 参数 fConfig CFilterConfig 的 对 象 ) 调用 
FilterConfig 的 getInitParameter(String paramName) 方 法 获得 参数 值 。 例 如 : 

public void init(FilterConfig fConfig) throws ServletException{ 

String en 一 {Config. getInitParameter( " encoding") ; 

) 

2) 一 filter-mapping 二 标记 及 其 子 标记 

web. xml 文件 中 出 现 一 个 二 filter 二 标记 就 会 对 应 地 出 现 一 个 二 filter-mapping 二 标记 。 
< 一 filter-mapping 过 标记 中 也 有 两 个 子 标 记 : < filterrname > HI < url-pattern =, Hf 
< 一 filter-name 二 子 标记 的 内 容 是 Web 服务 器 创建 的 filter 对象 的 名 字 (该 名 字 必 须 和 
< 一 filter 过 标记 的 子 标记 二 filtername 之 的 内 容 相 同 ); 过 url-pattern 二 子 标记 用 来 指定 用 户 
用 怎样 的 模式 请 求 filter 对 象 。 

如 果 有 某 个 URL 或 servlet 会 应 用 多 个 过 滤器 , 则 根据 一 filtermapping 之 标记 在 web 
.xml 中 出 现 的 先后 顺序 执行 过 滤器 。 

2. 运行 filter 

只 要 用 户 请 求 的 URL I —filter-mapping ff] FERI — url-pattern- 48 KE fff Ei 3X UG M, 


第 9 章 开发 Web 应 用 过 滤器 189 


Web 服务 器 就 会 自动 调用 该 filter 的 doFilter() 方 法 。 如 9. 1. 2 节 中 的 MyFirstFilter 过 滤 
器 在 web. xml 中 的 二 url-pattern 过 指定 值 为 /* ,“/ * ”代表 任何 页 面 或 servlet 的 请 求 。 


9.2.3 拓展 训练 


按照 本 节 的 任务 内 容 将 9.1.2 节 中 的 过 滤器 MyFirstFilter 部 署 成 功 ,并 运行 Web 应 
用 程序 测试 该 过 滤器 。 


9.3 创建 Web 应 用 过 滤器 


9.3.1 知识 要 点 


在 Web 工程 中 , 某 些 页 面 或 servlet 只 有 用 户 登 录 成 功 才 能 访问 。 直 接 在 应 用 程序 每 
个 相关 的 源 代码 中 判断 用 户 是 否 登录 成 功 ,这 并 不 是 科学 的 做 法 。 我 们 可 以 实现 一 个 登录 
验证 过 滤器 ,在 Web 工程 的 web. xml 中 设置 并 使 用 该 过 滤器 ,就 可 以 不 用 在 每 个 相关 的 源 
代码 中 验证 用 户 是 否 登录 成 功 。 


9.3.2 技能 操作 


创建 Web 应 用 之 登录 验证 过 滤器 。 具 体 任务 如 下 : 

新 建 一 个 Web 工程 loginValidate, 在 该 Web 工程 中 至 少 编写 两 个 JSP 页 面 (login. jsp 
与 loginSuccess. jsp) 和 一 个 servlet( 由 LoginServlet. java 负责 创建 )。 用 户 在 login. jsp 页 
面 中 输入 用 户 名 和 密码 后 ,提交 给 servlet, E servlet 中 判断 用 户 名 和 密码 是 否 正确 , 若 正 
确 , 则 跳 转 到 loginSuccess. jsp; 若 错误 , 则 回 到 login. jsp 页 面 。 但 该 Web 工程 有 另外 一 个 
要 求 : 除 了 访问 login. jsp 页 面 外 . 别 的 页 面 或 一 m > [http //localhost:6000/loginValidate/logim jsp 
servlet 都 不 能 直接 访问 ,必须 先 登 录 成 功 才能 访 


[| | 
问 。 我 们 在 设计 这 个 Web 工程 时 ,编写 了 一 个 登 p E 
录 验 证 过 滤器 并 在 该 Web 工程 中 使 用 。 一 | 
页 面 运行 效果 如 图 9-2 至 图 9-4 所 示 。 9-2 登录 画面 


L m d [http //localhost:8080/loginValidate/loginSuccess.jsp 
您 没有 登录 ， 请 先 登录 ! 3 种 钟 后 回 到 登录 页 面 。 
9-3 没有 登录 成 功 直接 运行 loginSuccess. jsp 


oco d http: //localhost:8080/loginValidate/loginSuccess. jsp 
恭喜 filter 登 录 成 功 ! 
图 9-4 登录 成 功 页 面 


190 网 络 编程 与 动态 网 站 制作 


代码 模板 web. xml 如 下 : 


<web-app> 


<filter> 
<filter-name> loginValidateFilter< /filter-name> 
<filter-class> filters. LoginFilter< /filter-class> 
<init-param> 
<param-name> login_uri</param-name> 
<param-value>/ login. jsp</param-value> 
</init-param> 
<init-param> 
<param-name> login_Servlet</param-name> 
param-value> /isLogin</param-value> 
</init-param> 
</filter> 


<filter-mapping> 
<filter-name> loginValidateFilter< /filter-name> 
<url-pattern>/ * </url-pattern> 

< /filter-mapping> 


<servlet> 
<servlet-name> loginServlet< /servlet-name> 
servlet-class> servlet. LoginServlet< /servlet-class> 
</servlet2> 


<servlet-mapping> 
<servlet-name>_oginServlet< /servlet-name> 
<url-pattern>/isLogin</url-pattern> 
</servlet-mapping> 


</web-app> 
代码 模板 LoginFilter. java( 过 滤器 ) 如 下 : 


package filters; 

import java. io. IOException; 

import java. io. PrintWriter; 

import javax. servlet. Filter; 

import javax. servlet. FilterChain; 

import javax. servlet. FilterConfig; 

import javax. servlet. ServletException ; 

import javax. servlet. ServletRequest ; 

import javax. servlet. ServletResponse; 

import javax. servlet. http. HttpServletRequest; 
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import javax. servlet. http. HttpServletResponse; 
import javax.servlet. http. HttpSession; 
public class LoginFilter implements Filter { 


private String logon. page; // 登 录 页 面 
private String logon_servlet; // 登 录 servlet 请 求 
// 消 灭 filter 方 法 


public void destroy() { 
} 
// 过 滤器 服务 方法 
public void doFilter( ServletRequest request, ServletResponse response, 
FilterChain chain) throws IOException, ServletException { 
HttpServletRequest req = (HttpServletRequest) request; 
HttpServletResponse resp = (HttpServletResponse) response; 
resp. setContentType( "text/html;"); 
resp. setCharacterEncoding("GBK"); 
HttpSession session = req. getSession(); 
PrintWriter out = resp. getWriter() ; 
// 得 到 用 户 请 求 的 URI 
String request uri = req. getRequestURI(C) ; 
// 得 到 Web 应 用 程序 的 上 下 文 路 径 
String ctxPath = req. getContextPath() ; 
// 去 除 上 下 文 路 径 , 得 到 剩余 部 分 的 路 径 
String uri = request uri. substring(ctxPath. length()); 
// 判断 用 户 访问 的 是 否 是 登录 页 面 或 提交 登录 请 求 
if (uri. equals(logon, page) | | uri. equals(logon_servlet)) { 
// 执 行 下 一 个 过 滤器 


chain. doFilter(request, response); 


) else ( 
// 如 果 访 问 的 不 是 登录 页 面 , 则 判断 用 户 是 否 已 经 登录 
if (null != session. getAttribute("user") 


8.8. "" I= session. getAttribute("user")) { 
// 执 行 下 一 个 过 滤器 
chain. doFilter(request，response) ; 
) else ( 
out. println(" 您 没有 登录 ,请 先 登 录 !3 秒 钟 后 回 到 登录 页 面 ."); 
resp. setHeader( "refresh", "3;url=" + ctxPath + logon page); 
return; 


} 
} 
// 过 滤器 初始 化 方法 
public void init(FilterConfig config) throws ServletException { 
// 从 web. xml 的 部 署 描述 符 中 获取 登录 页 面 
logon page = config. getInitParameter("login_uri"); // 获 得 参数 login uri 的 值 
logon servlet = config. getInitParameter("login Servlet"); // 获 得 参数 login uri 的 值 
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代码 模板 LoginServlet. java 如 下 : 


package servlet; 
import java.io. IOException; 
import javax. servlet. ServletException; 
import javax. servlet. http. HttpServlet; 
import javax. servlet. http. HttpServletRequest; 
import javax. servlet. http. HttpServletResponse; 
import javax. servlet. http. HttpSession; 
public class LoginServlet extends HttpServlet | 
protected void doGet( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
String username- request. getParameter( " name") ; 
String password — request. getParameter( "pwd") ; 
if(username! = null&.& username. equals( " filter") ) ( 
if( password! = null&- & password. equals( "filter" )) ( 
HttpSession session request. getSession( ) ; 
session. setAttribute("user", username); 
response. sendRedirect( "loginSuccess. jsp") ; 
Jelse( 
response. sendRedirect( "login. jsp") ; 
) 
Jelse( 
response. sendRedirect( "login. jsp") ; 


) 
protected void doPost( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
doGet(request, response) ; 


) 
代码 模板 login. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 
<title> login. jsp</title> 
</head> 
<body bgcolor= "lightPink"> 
<form action= "isLogin" method= "post" > 
<table> 
<tr> 
«ud JH PA: </td> 
«td input type="text" name= "name" /7 « /td> 
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</tr> 

<tr> 
二 td 二 密码: </td> 
<td><input type="password" name= "pwd"/></td> 

</tr> 

<tr> 
<td><input type="submit" value= "1 22" /7 — / td 
<td><input type— "reset" value= " 8 $t" /7 — /td> 

</tr> 

</table> 
</form> 
<body> 
</html> 


代码 模板 loginSuccess. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK" 42 
<html> 
<head> 
<title>loginSuccess. jsp< /title> 
</head> 
<body> 
<% 
String username= (String)session. getAttribute("user") ; 
%> 
恭喜 一 % — username Yo XE SE JR JI ! 
</body> 
</html> 


上 述 任务 中 的 过 滤器 ,要 首先 检查 用 户 请 求 的 URL 是 不 是 login. jsp 或 者 登录 请 求 
(isLogin) ,这 两 个 值 都 放 在 了 过 滤器 的 初始 化 参数 中 。 如 果 用 户 访问 的 是 login. jsp 或 者 登 
录 请 求 ,过 滤器 就 执行 chain. . doFilter() 继 续 请 求 。 如 果 用 户 访问 的 不 是 login. jsp 或 者 登 
录 请 求 , 则 过 滤器 先 判 断 用 户 是 否 登录 成 功 ,登录 成 功 则 执行 chain... doFilter() 继 续 请 求 ， 
否则 重 定向 到 login. jsp. 


9.3.3 拓展 训练 


1. 简 述 过 滤器 的 运行 原理 。 

2. Filter 接口 中 有 哪些 方法 ? 它们 分 别 具 有 哪些 功能 ? 

3. 在 web. xml 中 部 署 过 滤器 需要 哪些 标记 ? 这 些 标记 的 作用 是 什么 ? 

4. 在 任务 的 Web 工程 loginValidate 中 再 新 建 几 个 JSP 页 面 ,在 没有 登录 成 功 的 情况 
运行 


下 ,运行 这 几 个 JSP 页 面 看 看 是 什么 效果 。 
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9.4 小 结 


* 在 JSP/Servlet 要 实现 过 滤器 ,必须 实现 Filter 接口 ,并 在 web. xml 中 定义 部 署 过 滤 
器 ,让 服务 器 知道 加 载 哪个 过 滤器 。 

* Filter 接口 有 init() ,doFilter() 与 destroyO 三 个 方法 。 这 三 个 方法 与 Servlet 接口 
的 initO ,serviceO 5j destroy() 类 似 。 

* 过 滤器 必须 在 web. xml "P AB. a] UI ffi JH < filter fI < filter-mapping fx ie 8p A 
ibükse. Hor HP —filter-name- WEE PF 08 $8 I6 44 FI. f JH — filter-class BB iE UE 
器 的 类 名 e fili HI url-pattern = Ap URL 的 请 求 模式 。 
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表达 式 语言 


在 前 面 章节 中 编写 JSP 页 面 时 ,经 常 使 用 Java 代码 来 实现 页 面 显 示 逻 辑 。 网 页 中 夹杂 
着 HTML 与 Java 代码 ,给 网 站 的 设计 与 维护 带 来 困难 。 我 们 可 以 使 用 表达 式 语言 
(Expression Language，EL) 来 访问 和 处 理应 用 程序 的 数据 。 这 样 JSP 页 面 就 尽量 减少 了 
Java 代码 的 使 用 ,为 以 后 的 工作 提供 了 方便 。 本 章 将 重点 介绍 表达 式 语言 EL 的 基本 用 法 。 


10.1 使 用 EL 访问 对 象 的 属性 


10.1.1 知识 要 点 


EL 是 JSP 2. 0 规范 中 增加 的 , 它 的 基本 语法 为 : 

$ {表达 式 } 
类 似 于 JSP AGAR 1⁄4 = RKRN >, EL 语句 中 的 表达 式 值 会 被 直接 送 到 浏览 器 显示 。 
使 用 EL 可 以 获取 对 象 的 属性 ,如 JavaBean, 数组 或 List 对 象 。 

1. 获取 JavaBean 的 属性 值 

假设 在 JSP 页 面 中 有 这 样 一 条 语句 : 


<jsp:getProperty property= "age" name= "user" />2> 
那么 ,可 以 使 用 EL 获取 user 的 属性 age, 修 改 如 下 ; 
$ (user. age} 


其 中 ,点 运算 符 前 面 为 JavaBean 的 对 象 user, 后 面 为 该 对 象 的 属性 age, 表 示 利 用 user 对 象 
的 getAge() 方 法 取得 值 ,而 后 显示 在 网 页 上 。 
2. 获取 数组 中 的 元 素 
假设 在 JSP 页 面 中 有 这 样 一 段 代 码 : 
<% 
String dogs[] = (" lili", "huahua", "guoguo") ; 


request. setAttribute("array", dogs): 
> 


那么 ,在 页 面 某 处 可 以 使 用 EL 取出 数组 中 的 元 素 ,代码 如 下 : 
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$ (array[0]) 
$ (array[1]) 
$ (array[2]) 


3. 获取 List 对 象 中 的 元 素 
假设 在 JSP 页 面 中 有 这 样 一 段 代码 : 


<% 
ArrayList< UserBean- users=new ArrayList< UserBean>() ; 
UserBean ubl = new UserBean( "zhang" , 20); 
UserBean ub2= new UserBean( "zhao" , 50) ; 
users. add(ub1) ; 
users. add(ub2) ; 
request. setAttribute("array", users); 


> 
其 中 ,UserBean 有 两 个 属性 : name 和 age, 那 么 在 页 面 某 处 可 以 使 用 EL 取出 UserBean 中 
的 属性 ,代码 如 下 : 


$ (array[0].name) $ (array[0] . age} 
$ {array[1] . name} $ {array[1] . age} 


10.1.2 技能 操作 


使 用 EL 表达 式 取 出 对 象 的 属性 。 具 体 任 务 如 下 : 

编写 一 个 创建 JavaBean 对 象 的 类 UserBean, 该 类 中 有 两 个 属性 (成 员 变 量 ): name 和 
age, 然 后 在 JSP 页 面 eg10_1. jsp 中 使 用 EL 取出 该 JavaBean 对 象 的 属性 。 

代码 模板 UserBean. java 如 下 : 


package bean; 
public class UserBean ( 
String name; 
int age; 
public UserBean() | 
name 一 "EL 表达 式 "; 
age 一 5; 
) 
public String getName() { 
return name; 
) 
public void setName(String name) ( 
this. name — name; 
) 
public int getAge() { 
return age; 
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} 
public void setAge(int age) { 
this.age = age; 
) 
) 


代码 模板 eglO l.jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<%@ page import= "bean. * " %> 
<html> 
<head> 
<title> EL i5 X — /title> 
</head> 
<body> 
<jsp:useBean id= "user" class= "bean. UserBean" scope= "page" /2> 
姓名 : $ {user.name) —!-- 使 用 EL 取出 name 属性 --> 
<br> 


年 龄 : $ (user.age) <!-- 使 用 EL 取出 age 属性 --> 

</body> 

</html> 

代码 分 析 : 

* EL 处 理 null 值 。 

对 于 null 值 直接 以 空 字符 串 显 示 ,而 不 是 null, 运 算 时 也 不 会 发 生 错误 或 空 指针 异常 。 
所 以 在 使 用 EL 访问 对 象 的 属性 时 ,不 需要 判断 对 象 是 否 为 null 对 象 。 这 样 就 为 编写 程序 
提供 了 方便 。 

* JSP 页 面 与 EL, 

JSP 页 面 默认 支持 EL, 但 如 果 JSP 页 面 使 用 page 指令 设置 isELIgnored 属性 (默认 为 
false) 值 为 true, 则 该 页 面 不 能 使 用 EL. 

。 对 象 的 有 效 范 围 。 

在 EL 中 ,可 以 使 用 EL 内 置 对 象 指定 范围 来 访问 属性 ,EL 内 置 对 象 将 在 稍 后 介绍 。 
如 果 不 指 定 对 象 的 有 效 范围 , 则 以 page、request、session、application 的 顺序 查找 EL 中 所 指 
定 的 对 象 。 

* (. ) 与 [] 运 算 符 的 区 别 。 

EL 中 点 运算 符 (. ) 和 [j 运 算 符 ,在 一 些 情况 下 用 法 是 一 样 的 ,总 结 如 下 : 

OD. C ) 运 算 符 左边 可 以 是 JavaBean 或 Map 对 象 。 

(2) [] 运 算 符 左边 可 以 是 JavaBean、Map、 数 组 或 List 对 象 。 

使 用 EL 如 何 取得 Map 对 象 中 的 值 呢 ? 假设 在 JSP 页 面 中 有 这 样 一 段 代码 : 
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<% 
HashMap< String, String> map 一 new HashMap< String, String> () ; 
map. put("fisrt", "55—"); 
map. put("second", "$5 —"); 
request. setAttribute(" number", map): 
%> 


那么 在 页 面 某 处 可 以 使 用 EL 获得 Map 中 的 值 ,代码 如 下 : 


$ (number. fisrt) 
$ (number. second) 


$ (number["fisrt"]) 
$ (number["second"]) 
10.1.3 拓展 训练 
如 果 把 上 述 eg10_1. jsp 中 的 代码 : 


<jsp:useBean id= "user" class= "bean. UserBean" scope= "page" />> 


删除 ,然后 再 运行 程序 ,查看 页 面 显示 结果 ; 如 果 不 使 用 EL 取 值 ,而 使 用 标记 所 jsp， 
getProperty 二 取 值 ,那么 再 删除 


<jsp:useBean id= "user" class= "bean. UserBean" scope= "page" /2> 


的 情况 下 ,运行 网 页 会 是 什么 结果 ? 
10.2 EL 内 置 对 象 


EL 内 置 对 象 共 有 11 个 ,本 节 只 是 介绍 几 个 常用 的 EL 内 置 对 象 ， pageScope、 
requestScope, sessionScope ,applicationScope, param 以 及 param Values, 
10.2.1 知识 要 点 

1. 与 作用 范围 相关 的 内 置 对 象 

与 作用 范围 相关 的 EL 内 置 对 象 有 pageScope、 requestScope、sessionScope 和 

applicationScope, 分 别 可 以 获取 JSP 内 置 对 象 pageContext, request, session 和 application 
中 的 数据 。 如 果 在 EL 中 没有 使 用 内 置 对 象 指定 作用 范围 , 则 从 作用 范围 为 pageScope 的 
数据 开始 寻找 。 获 取 数据 的 格式 如 下 : 

$ (EL 内 置 对 象 . 关键 字 对 象 .属性 } 
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或 
$ (EL 内 置 对 象 .关键 字 对 象 } 
例如 : 


user" class= "bean. UserBean" scope="page"/> 


<jsp:useBean id=" 
<jsp:setProperty name= "user" property= "name" value= "EL 内 置 对 象 " /二 


name: $ {pageScope. user. name} 


再 比如 ,在 JSP 页 面 中 有 这 样 一 段 代 码 : 


<% 
ArrayList< UserBean> users=new ArrayList< UserBean > () ; 
UserBean ubl =new UserBean( "zhang" , 20) ; 
UserBean ub2= new UserBean( "zhao" , 50) ; 
users. add(ub1) ; 
users. add(ub2) ; 
request. setAttribute("array", users); 


n 


其 中 ,UserBean 有 两 个 属性 : name 和 age, JÉ Z TE request 有 效 的 范围 内 可 以 使 用 EL 取出 


UserBean 的 属性 ,代码 如 下 : 


$ (requestScope.array[0].name) $ (requestScope.array[0] . age) 
$ (requestScope.array[1].name) $ (requestScope.array[1] . age) 


2. 与 请 求 参数 相关 的 内 置 对 象 


表达 式 


m= 


Wt ri 


与 请 求 参 数 相 关 的 EL 内 置 对 象 有 param 和 paramValues。 获 取 数 据 的 格式 如 下 : 


$ (EL 内 置 对 象 .参数 名 } 
例如 ,input. jsp 的 代码 如 下 : 


<form method = "post" action = "param.jsp"> 
p l4: <input type="text" name= "username" size="15" /></p> 
pd: 


<input type="checkbox" name= "habit" value "f 45" /7 8 E 
<input type= "checkbox" name= "habit" value 一 " 玩 游戏 "/ 二 玩 游戏 
<input type="checkbox" name= "habit" value 二 "旅游 "/ 二 旅游 
<p> 
<input type="submit" value= "1 22 "/7— 

</form> 


那么 ,在 param. jsp 页 面 中 可 以 使 用 EL 获取 参数 值 , 代 码 如 下 : 


< request. setCharacterEncoding("GBK") ; ⁄> 
<body> 
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<h2>EL KAX param, paramValues</h2> 
姓名 : $ (param. username) </br> 

兴趣 : 

$ {paramValues. habit[0]) 

$ {paramValues. habit[1]) 

$ {paramValues. habit[2]) 


10.2.2 技能 操作 


使 用 EL VJ HOS 8A JSP 内 置 对 象 中 获取 数据 。 具 体 任务 如 下 : 

编写 一 个 Servlet 类 ,在 该 类 中 使 用 request 内 置 对 象 存储 数据 ,然后 从 该 servlet 转发 
到 show. jsp 页 面 ,最 后 在 show. jsp 页 面 中 显示 request 内 置 对 象 的 数据 。 首先 ,运行 
servlet, 在 IE 地 址 栏 中 输入 : 

http://localhost:8080/ch10/saveServlet 


程序 运行 结果 如 图 10-1 所 示 。 


@ ` 8 P [http//localhost.8080/ch10/saveServlet 
从 servlet 转 发 过 来 的 request 内 置 对 象 的 数据 如 下 ， 
zhao 


qian 
sun 
li 


10-1 使 用 EL 内 置 对 象 获取 JSP 内 置 对 象 的 数据 
配置 文件 web. xml 如 下 : 


<servlet> 
<servlet-name> saveServlet</servlet-name> 
<servlet-class> servlet. SaveServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name> saveServlet</servlet-name> 
url-pattern> /saveServlet< /url-pattern> 
< /servlet-mapping> 


代码 模板 SaveServlet. java 如 下 : 


package servlet; 

import java. io. IOException; 

import javax. servlet. RequestDispatcher; 
import javax. servlet. ServletException; 

import javax. servlet. http. HttpServlet; 

import javax. servlet. http. HttpServletRequest; 
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import javax. servlet. http. HttpServletResponse; 
public class SaveServlet extends HttpServlet ( 
protected void doGet( HttpServletRequest request, HttpServletResponse response) 
throws ServletException, IOException | 
String names[] = ("zhao" , "gian" , "sun", "li") ; 
request. setAttribute("name", names); 
RequestDispatcher dis— request. getRequestDispatcher( "show. jsp") ; 
dis. forward(request, response); 
) 
protected void doPost( HttpServletRequest request, HttpServletResponse response) 
throws ServletException, IOException ( 
doGet(request, response) ; 


) 
代码 模板 show. jsp iH F; 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<html> 
<head> 

<title> EL HEH $$ — /title 
</head> 
<body> 

从 servlet 转发 过 来 的 request 内 置 对 象 的 数据 如 下 : <br> 
$ (requestScope. name[0]) <br> — !--f# Hj EL 内 置 对 象 requestScope 取出 request 中 数组 的 第 1 个 
元 素 -->> 
$ (requestScope. name[1]) br — !--## H] EL 内 置 对 象 requestScope 取出 request 中 数组 的 第 2 个 
XX— 
$ (requestScope. name[2] ) br — !--fi H] EL P3 XE $& requestScope 取出 request 中 数组 的 第 3 个 
TZ--> 
$ (requestScope. name[3] ) br — !--f H] EL 内 置 对 象 requestScope 取出 request 中 数组 的 第 4 个 
XX 
</body> 
</html> 


通过 上 面 的 例子 可 以 看 出 ,EL AEIR JSP 内 置 对 象 不 同 ,EL 内 置 对 象 仅仅 代表 
作用 范围 。 
10.2.3 拓展 训练 

1. 在 Web 应 用 程序 中 有 以 下 程序 代码 段 ,执行 后 转发 到 某 个 JSP SUR: 


ArrayList<String> dogNames= new ArrayList< String> () ; 
dogNames. add( "goodDog") ; 
request. setAttribute("dogs", dogNames) ; 
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以 下 ) 选 项 可 以 正确 地 使 用 EL 取得 数组 中 的 值 。 


A. $1{ dogs .0} B. ${ dogs [0]} 

C. ${ dogs .[0]} D. ${ dogs "0"} 
2. C ORE EL 的 内 置 对 象 。 

A. request B. pageScope 

C. sessionScope D. applicationScope 


3. 如 果 把 任务 中 show. jsp 页 面 中 的 代码 : 


$ (requestScope. name[0] ) br 
$ (requestScope. name[1] ) br 
$ (requestScope. name[2] ) br 
$ (requestScope. name[3] ) br 


改 成 : 


$ (name[0]) <br> 
$ (name[1]) <br> 
$ (name[2]) br 
$ (name[3]) br 


然后 运行 程序 ,查看 运行 结果 。 
10.3 小 结 


。 在 JSP 页 面 中 一 些 简单 的 属性 \ 请 求 参 数 等 值 的 获取 ,一 些 简 单 的 运算 或 判断 ,可 以 
使 用 EL 表达 式 来 处 理 , 减 少 了 页 面 中 的 Java 代码 。 

。 可 以 使 用 EL 表达 式 获 取 对 象 的 属性 值 ,例如 JavaBean X1 $£ Map 对 象 、 数 组 或 List 
对 象 , 还 可 以 使 用 它 获取 JSP 内 置 对 象 中 的 数据 。 


第 11 章 


标准 标签 库 


在 网 站 开发 与 制作 过 程 中 ,可 以 使 用 标准 标签 库 (JavaServer Pages Standard Tag 
Library,JSTL) 来 替换 网 页 中 实现 页 面 显 示 逻 辑 的 Java 代码 。 这 样 也 可 以 减少 JSP 页 面 中 
Java 代码 的 使 用 ,为 后 续 的 维护 工作 提供 方便 。 本 章 将 重点 介绍 标准 标签 库 JSTL 的 基本 
用 法 。 


11.1 一 般 用 途 的 标签 


JSTL 是 一 个 标准 规范 ,但 不 在 JSP 的 规范 中 ,所 以 需要 下 载 JSTL 实现 (jar 包 )。 可 以 
登录 网 站 

https://jstl. dev. java. net/ 
下 载 JSTL1.2 的 jar 包 : jstl-impl-1. 2.jar。 另 外 ,还 需要 JSTL 标准 接口 与 类 (jstl. jar)。 

如 果 使 用 Tomcat 作为 Web 服务 器 ,可 以 在 Tomcat. 的 webapps\examples\WEB-INF\lib 
中 找到 jstl. jar 文件 。 在 JSP 页 面 中 要 想 使 用 JSTL 核心 标签 库 , 必 须 把 jstl-impl-1. 2. jar 
5j jstl. jar 复制 到 Web 工程 的 WEB-INF\lib 中 。 同 时 在 JSP 页 面 中 使 用 taglib 标记 定义 
前 置 名 称 与 uri 引用 ,代码 如 下 : 


<%@ taglib prefix— "c" uri="http://java. sun. com/jsp/jstl/core" Yi > 

本 书 中 只 说 明 JSTL 核心 标签 库 中 几 个 常用 的 标签 ,其 他 标签 请 参考 JSTL 说 明文 档 
或 专门 的 书籍 。 
11.1.1 知识 要 点 


1. 一 c:out 二 标签 
去 c:out> 用 来 显示 数据 的 内 容 ,与 二 %% 一 表达 式 汉 > 或 ${ 表 达 式 } 类 似 。 格 式 如 下 ， 


<c:out value 一 "输出 的 内 容 " [default= "defaultValue"]/> 


<c:out value= "$ th f] P E" 
defaultValue 
</c:out> 
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其 中 ,value 值 可 以 是 一 个 EL 表达 式 , 也 可 以 是 一 个 字符 串 ; default 可 有 可 无 , 当 value 值 
不 存在 时 ,就 输出 defaultValue。 例 如 : 

<c:out value=" $ ( param. data)" default= "No Data" /> 

<br> 

—c:out value=" $ (param.nothing) " /> Bl > [http://ocalhost:8080/ch11/NewFile jsp 

<br> 

<c:out value= "This is a String" /> 

输出 的 结果 如 图 11-1 所 示 。 

2. —eset^ r 图 11-1 <c;out> 453 

。 设置 作用 域 变量 。 

H AEH <c:set > page, request, session application 等 范围 内 设置 一 个 变量 。 一 般 
格式 如 下 : 


No Data 


This is a String 


<c:set value= "value" var 一 "varName"”[scope 一 "page|request| session application"] /之 
将 value 值 赋值 给 变量 varName。 例 如 : 
<c:set value— "zhao" var= "userName" scope= "session" />> 
相当 于 
<% session. setAttribute( "userName", "zhao"); ⁄> 


* 设置 JavaBean 的 属性 。 
使 用 一 c:set 二 设置 JavaBean 的 属性 时 ,必须 使 用 target 属性 进行 设置 。 其 格式 如 下 : 


<c:set value= "value" target= "target" property= “propertyName"/> 


将 value 赋值 给 target 对 象 (JaveBean 对 象 ) propertyName 属性 。 如 果 target 为 null 或 
没有 set 方法 则 抛 出 异常 。 

3. <c: remove > pk Z 

如 果 要 删除 某 个 变量 T E <c: remove tR. WIJANG: 


<c:remove var— "userName" scope= "session" />> 


相当 于 


< session. removeAttribute("userName") %> 


11.1.2 技能 操作 


灵活 使 用 JSTL 基本 输入 输出 标签 ,具体 任务 如 下 : 
编写 一 个 JSP 页 面 input. out. jsp ,在 该 页 面 中 使 用 过 c:set 之 标签 定义 几 个 变量 ,并 使 
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用 二 c:out> 标 签 输出 这 几 个 变量 的 值 。 运 行 效果 如 图 11-2 所 示 。 


Æ > [http://ocalhost:8080/ch11/input_outjsp 


varl :setåndout 
var2:1+2 
var3:3 

var4:4 


11-2 <c:set> 5 ciout^ KRA 
代码 模板 input_out. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset= GBK" 
pageEncoding= "GBK" 4 > 

<%@ taglib prefix— "c" uri— "http: //java. sun. com/jsp/jstl/core" % > 
<html> 

<head> 

<title>input_out. jsp</title> 

</head> 

<body> 

<c:set var="varl" value= "setAndout" /> 

<c:set var="var2" value="1+2"/> 

<c:set var="var3" value=" $ (1+2)" /> 

<c:set var="var4 scope ="request" value=" $ {1 + 2}" /> 
varl:<c:out value=" $ (var1) " default= "No Data" /><p> 

$ {var2}" default= "No Data" /><p> 
var3:<c:out value=" $ {var3}" default="No Data" /><p> 
var4:<c:out value=" $ {var4+1}" default= "No Data" /><p> 
</body> 

</html> 


在 10. 2 节 中 使 用 EL 表达 式 就 可 以 输出 变量 或 表达 式 的 值 ,那么 我 们 为 什么 还 要 学 习 
去 c:out> 标 签 呢 ? 下 面 先 来 猜 猜 这 段 程序 的 输出 结果 是 什么 ? 
<% 
String s=" <p> H RRF </p>"; 
request. setAttribute("exp", s); 
%> 
$ {exp} 
运行 时 才 发 现 其 中 的 HTML 标记 二 p 记 没有 起 到 创建 段落 的 作用 。 如 果 希 望 二 p 二 达 
到 创建 段落 的 作用 ,那么 必须 把 语句 : 


var2: <c:out value 


$ (exp) 
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改 成 : 
<c:out value— "— p^ Kik EM </p>" escapeXml= "false" /这 


默认 情况 下 ,一 c:out 二 将 所、 >、 M & 转换 为 &lt;、&-gt;、& # 039; & # 034; 和 
&amp; ,如 果 不 想 转换 ,只 需 将 escapeXml 属性 设置 为 false。 


11.1.3 拓展 训练 
把 前 述 任务 中 input. out. jsp 页 面 里 的 代码 : 


varl:<c:out value=" $ (var1) " default= "No Data" /><p> 
var2:-c:out value=" $ (var2) " default= "No Data" /><p> 
var3: <c:out value=" $ (var3) " default= "No Data" /><p> 
var4:<c:out value=" $ (var4+ 1)" default= "No Data" /><p> 


改 成 : 


varl: $ {varl}<p> 
var2: $ (var2) <p> 
var3: $ (var3) <p> 
var4: $ (var4+1) <p> 


然后 再 运行 程序 ,查看 结果 有 什么 不 同 。 
11.2 条 件 控 制 标 签 


11.2.1 知识 要 点 


1. 一 c:if 一 标签 

去 c:if 之 标签 实现 话语 句 的 作用 ,具体 语法 格式 如 下 : 

«cf test 二 "条 件 表达 式 " 二 

主体 内 容 

</c:if> 
其 中 ,条 件 表达 式 可 以 是 EL 表达 式 , 也 可 以 是 JSP 表达 式 。 如 果 表 达 式 的 值 为 true, 则 会 
执行 二 c:if 二 的 主体 内 容 , 但 是 没有 相对 应 的 二 c:else 二 标签 。 如 果 想 在 条 件 成 立时 执行 一 
块 内 容 , 不 成 立时 执行 男 一 块 内 容 , 则 可 以 使 用 二 c:choose 记 ,二 c: when = K < c: otherwise 
标签 。 

2. <e: choose> , <c; when> Jk — c otherwise > PF 22 

<c:choose> , <c: when> K <c:otherwise> fj && KH if/else if/else 语句 的 作用 。 具 
体 语 法 格式 如 下 : 
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<c:choose> 
<c:when test 二 "条 件 表达 式 1"> 
主体 内 容 1 
</c:when> 
<c:when test= " A fF Xik ok 2"> 
主体 内 容 2 
</c:when> 
<c:otherwise> 
表达 式 都 不 正确 时 ,执行 的 主体 内 容 
</c:otherwise> 
</c:choose> 


11.2.2 技能 操作 


A <c if>, 一 c:choose 二 一 c:when 二 及 一 c:otherwise 二 标签 的 使 用 方法 。 具 体 任 
SF: 

编写 一 个 JSP 页 面 ifelse. jsp, 在 该 页 面 中 使 用 二 c:set 二 标签 把 两 个 字符 串 设置 在 
request 内 置 对 象 中 。 使 用 二 c:if 二 标签 求 出 这 两 个 字符 串 的 最 大 值 ( 按 字 典 顺序 比较 大 
小 ) ,使 用 一 c:choose 二 .一 c:when 二 及 一 c:otherwise 二 标签 求 出 这 两 个 字符 串 的 最 小 值 。 

代码 模板 ifelse. jsp 如 下 : 


<%@ page language= "java" contentType= "text/html; charset= GBK" 
pageEncoding= "GBK" %> 
<%@ taglib prefix— "c" uri="http://java. sun. com/jsp/jstl/core" Yi > 
<html> 
<head> 
<meta http-equiv= "Content-Type" content= "text/html; charset— ISO-8859-1"> 
<title>ifelse.jsp</title> 
</head> 
<body> 
<c:set value= "if" var= "firstNumber" scope= "request"/> 
<c:set value= "else" var= "secondNumber" scope= "request" />> 
<c:if test=" $ {firstNumber> secondNumber} "> 

最 大 值 为 $ (firstNumber) 
</c:if> 
<c:if test=" $ (firstNumber<secondNumber) "> 

最 大 值 为 $ (secondNumber) 
X/cif 
<c:choose> 

<c:when test= " $ (firstNumber<secondNumber) "> 

最 小 值 为 $ (firstNumber) 
</c:when> 
<c:otherwise> 


最 小 值 为 $ (secondNumber) 
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</c:otherwise> 
</c:choose> 
</body> 
</html> 


为 了 更 好 地 理解 本 例 中 的 代码 ,将 注意 事项 说 明 如 下 。 

<c:when> K — c:otherwise2 WA BOE — c: choose Zr, <c: when = WI test 结 
RH true 时 ,会 输出 二 c: when > fj d: IK V @ , Vi AS BB Z: < c: otherwise > fff VJ 7E. 
<c: choose> PAALA: when = ,程序 会 从 上 到 下 进行 条 件 判 断 ,如 果 有 个 二 c: when = hy 
test 结果 为 true, 就 输出 其 主体 内 容 , 之 后 的 二 c: when 二 就 不 再 执行 。 如 果 所 有 的 
<c:when> ff test 结果 都 为 false, 则 会 输出 二 c: otherwise MIWA, <c if> 5 
<c: choose> th nf UREE H AAN: 


<c:set value= "fda" var= "firstNumber" scope= "request"/> 
<c:set value= "else" var= "secondNumber" scope= "request" /> 
<c:set value="ddd" var— "threeNumber" scope= "request"/> 


<c:if test=" $ (firstNumber> secondNumber) "> 


<c: choose> 
<c:when test=" $ {firstNumber< threeNumber} "> 
最 大 值 为 $ (threeNumber) 
</c:when> 


<c:otherwise> 
最 大 值 为 $ {firstNumber} 
</c:otherwise> 
</c:choose> 
«cf 


<c:if test—" $ (secondNumber- firstNumber) "> 
<c:choose> 
<c:when test=" $ (secondNumber<threeNumber) "> 
最 大 值 为 $ (threeNumber) 
</c:when> 
<c:otherwise> 
最 大 值 为 $ (secondNumber) 
</c:otherwise> 
</c:choose> 
</c:if> 


11.2.3 拓展 训练 


编写 一 个 JSP 页 面 useTag. jsp, 在 该 页 面 中 使 用 二 c:set 二 标签 把 三 个 字符 串 设 置 在 
request W EX ZP. WH<: if>, 一 c:when 二 、 一 c:otherwise 二 和 二 c:choose 二 标签 求 出 


这 三 


11. 


个 字符 串 的 最 小 值 。 


3.1 知识 要 点 


如 下 : 


<c:forEach var 一 "变量 名 ”items 一 "数组 或 Collection X4 # "> 
循环 体 
</c:forEach> 
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JSTL 的 一 c:forEach 之 标签 可 以 实现 程序 中 的 for 循环 ,完成 迭代 功能 。 语 法 格式 


其 中 ,items 属性 可 以 是 数组 或 Collection 对 象 ,每 次 循环 读 取 对 象 中 的 一 个 元 素 , 并 赋值 给 
var 属性 指定 的 变量 ,之 后 就 可 以 在 循环 体 使 用 var 指定 的 变量 获取 对 象 的 元 素 。 例 如 ,在 
JSP 页 面 中 有 这 样 一 段 代码 : 


<% 
ArrayList< UserBean2> users=new ArrayList- UserBean-^ () ; 
UserBean ubl — new UserBean("zhao", 20) ; 
UserBean ub2= new UserBean( "qian" , 40) ; 
UserBean ub3 — new UserBean( "sun" , 60) ; 
UserBean ubi — new UserBean( "li", 80) ; 
users. add(ubl) ; 
users. add(ub2) ; 
users. add(ub3) ; 
users. add(ub4) ; 
request. setAttribute("usersKey", users) ; 


> 


JS s. t VC Hi E Ab nT PAM Hd <c: forEach> (891388 A £l Poe. RIAT : 


<table> 
<tr> 
<th> lE ch 
去 th> 年 龄 一 /th> 
</tr> 
<c:forEach var="user" items=" $ {requestScope. usersKey} "> 
<tr> 
<td> $ (user. name) </td> 
<td> $ (user. age) </td> 
</tr> 
</c:forEach> 
</table> 
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11.3.2 技能 操作 


使 用 一 c:forEach> 标 签 循环 遍历 集合 中 的 元 素 。 具 体 任务 如 下 : 
把 8. 2. 2 节 中 showAllGoods. jsp 页 面 里 的 for 语句 改 成 二 c:forEach 放 标签 。 
代码 模板 showAllGoods. jsp 如 下 : 


<% @ page language =" java" contentType =" text/html; charset = GBK" pageEncoding = 
"GBK"%> 
<%@ taglib prefix= "c" uri="http://java. sun. com/jsp/jstl/core" Yi > 
<%@ page import= "java. util. ArrayList" %> 
<%@ page import— "bean. Goods" %> 
<html> 
<head> 
<title>showAllGoods.jsp</title> 
</head> 
<body> 
<table border= "1"> 
<tr> 
<th> Wi infi </th> 
二 th 二 商品 名 称 二 /th 二 
二 th 二 商品 价格 二 /th 二 
三 th 二 商品 类 别 二 /th 二 
</tr> 
<c:forEach var= "good" items=" $ {requestScope. goods) "> 
<y> 
<td> $ (good. goodsld) </td> 
<td> $ (good. goodsName} </td> 
<td> $ (good. goodsPrice) </td> 
<td> $ (good. goodsType) </td> 
</tr> 
</c:forEach> 
</table> 
</body> 
</html> 


说 明 : 在 有 些 情 况 下 ,我 们 需要 为 二 c:forEach 二 标签 里 的 var 属性 指定 初始 值 
(begin) RIE Cend 和 步 长 (step)。 例 如 ,下 面 的 testTag. jsp 文件 : 


<table border— 177 


<tr> 
<th>number</th><th>Square</th> 
< /*r> 
<c:forEach var="x" begin="1" end—"5" step= "1"> 
<tr> 


<td><c:out value=" $ (x) "/></td> 
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<td><c:out value=" $ (x + x}"/></td> 
xh 
</c:forEach> 
</table> 


该 程序 运行 结果 如 图 11-3 所 示 。 


m > [http:/localhost8080/ch11/testTagjsp 


Square 


图 11-3 <c:forEach> iR% 


11.3.3 拓展 训练 
| € )JSTL 标签 可 以 实现 Java 程序 中 的 话语 句 功能 。 


A. <c;set> B. <c:out> C. <c:forEach> D. <c;if> 
2. )JSTL 标签 可 以 实现 Java 程序 中 的 for 语句 功能 。 
A. <c:set> B. <c:0ut> C. «eiforEach^ D. <c:if> 


3. 编写 一 个 JSP 页 面 mulTable. jsp. fg 9t ifii 11 fii Hd <c: forEach — bg 4 fi ih Ju Ju 3 
法 表 。 


11.4 小 结 


在 JSP 页 面 中 可 以 使 用 JSTL 标签 来 替代 实现 页 面 逻辑 的 Java EF. Øi, <c: if> 
替代 让 语句 和 二 c:forEach 二 替代 for 语句 。 


动态 网 站 开发 综合 实例 


本 章 通过 一 个 小 型 的 BBS 论坛 系统 实例 ,讲述 如 何 采 用 JSP 十 JavaBean 十 Servlet 的 模 
式 来 开发 一 个 简单 的 动态 网 站 。 系 统 将 业务 逻辑 封装 在 JavaBean 中 ,使 系统 的 可 维护 性 和 
可 扩展 性 大 大 提高 。 

系统 的 开发 环境 如 下 : 

* 操作 系统 Windows 7; 

* 数据 库 MySQL 5.5; 

* JSP 引擎 Tomcat 6.0; 

。 集成 开发 环境 (IDE)Eclipse IDE for JavaEE Developers. 


12.1 系统 分 析 与 设计 


12.1.1 系统 需求 分 析 


BBS(Bulletin Board System ,电子 公告 牌 系统 ) 俗 称 论坛 系统 ,是 互联 网 上 一 种 交互 性 
极 强 、 网 友 喜 闻 乐 见 的 信息 服务 形式 。 根 据 相 应 的 权限 ,论坛 用 户 可 以 进行 浏览 信息 、 发 布 
信息 、 回 复 信息 、 管 理 信息 等 操作 ,从 而 加 强 不 同 用 户 间 的 文化 交流 和 思想 沟通 。 

“BBS 论坛 系统 ”的 主要 功能 有 用 户 管理 ,版 块 管理 .帖子 管理 .友情 链接 管理 .广告 
管理 。 

其 中 各 主要 功能 大 体 说 明 如 下 : 

(1) 用 户 管理 主要 包括 用 户 注册 ,用 户 登 录 .用户 资料 修改 等 功能 。 

(2) 版 块 管理 主要 包括 增加 版 块 、 编 辑 版 块 、 删 除 版 块 等 功能 。 

(3) 帖子 管理 主要 包括 发 布 帖子 .回复 帖子 .搜索 帖子 .浏览 帖子 、 转 移 帖子 、 编 辑 帖子 、 
删除 帖子 .帖子 加 精 .帖子 置 顶 等 功能 。 

(4) 友情 链接 管理 主要 包括 增加 链接 、 修 改 链接 、 删 除 链接 等 功能 。 

(5) 广告 管理 主要 包括 放置 广告 .删除 广告 等 功能 。 

另外 ,需要 说 明 的 是 ,以 上 各 项 功能 中 有 些 只 需要 普通 用 户 权 限 就 能 够 完成 ,而 有 些 则 
需要 版 主 或 管理 员 权 限 才 能 完成 。 

结合 前 述 需 求 分 析 , 可 以 给 出 论坛 系统 的 总 体 结构 图 ,如 图 12-1 所 示 。 
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图 12-1 “BBS 论坛 系统 "总体 结构 图 


12.1.2 系统 功能 模块 划分 


注册 用 户 可 以 根据 权限 使 用 BBS 论坛 系统 进行 相应 的 操作 ,具体 的 系统 功能 模块 如 


表 12-1 所 示 。 
表 12-1 系统 功能 模块 划分 

模块 名 称 功能 描述 输入 内 容 输出 内 容 
用 户 注册 检测 注册 信息 用 户 名 等 注册 信息 注册 结果 (是 否 成 功 ) 
用 户 登录 合法 用 户 通过 验证 进入 系统 用 户 名 、 密 码 登录 状态 (是 否 成 功 ) 
用 户 资料 修改 | 根据 当前 状况 修改 个 人 信息 修改 的 信息 修改 信息 (是 否 成 功 ) 
发 布 帖子 根据 需要 发 布 帖子 用 户 的 言论 用 户 的 言论 
回复 帖子 回复 已 存在 的 话题 帖子 用 户 的 回复 用 户 的 回复 
搜索 帖子 根据 需要 搜索 帖子 搜索 条 件 符合 搜索 条 件 的 帖子 
浏览 帖子 浏览 任意 版 块 帖 子 选择 任意 话题 帖子 该 话题 帖子 及 其 回复 
转移 帖子 根据 实际 情况 移动 帖子 位 置 “移动 "命令 移动 结果 (是 否 成 功 ) 
编辑 帖子 修改 曾经 发 过 的 帖子 修改 的 内 容 修改 后 的 内 容 
删除 帖子 删除 违规 帖子 “删除 ”命令 删除 结果 (是 否 成 功 ) 
帖子 加 精 将 重要 话题 帖子 列 为 精华 帖子 “加 精 ” 命 令 添加 加 精 图 标的 帖子 
帖子 置顶 将 重要 话题 帖子 放置 于 最 上 方 “置顶 ”命令 添加 置顶 图 标的 帖子 
添加 版 块 添加 版 块 、 设 置 版 主 版 块 的 相关 信息 版 块 列表 
修改 版 块 修改 版 块 信息 版 块 的 修改 信息 修改 结果 (是 否 成 功 ) 
删除 版 块 删除 版 块 “删除 ”命令 删除 结果 (是 否 成 功 ) 
增加 链接 接受 友情 链接 申请 ,等 待 验证 友情 网 站 的 信息 友情 网 站 的 链接 
修改 链接 验证 并 修改 友情 链接 信息 友情 链接 信息 修改 后 的 友情 链接 
删除 连接 清理 不 合格 的 友情 链接 “删除 ”命令 删除 结果 (是 否 成 功 ) 
设置 广告 选择 已 有 位 置 发 布 广告 广告 语 .URL 地 址 前 台 广 告 
删除 广告 清理 已 发 布 的 广告 “删除 ”命令 原 有 的 广告 消失 
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12.2.1 


12.2 数据 库 设 计 


本 系统 利用 MySQL 5. 5 创建 icefish 数据 库 , 并 在 该 数据 库 中 创建 相应 的 数据 表 ; 在 
具体 实现 上 采用 加 载 纯 Java 数据 库 驱 动 程序 的 方式 连接 MySQL 5. 5 数据 库 。 


数据 库 浸 辑 结 构 设 计 


根据 系统 需求 分 析 和 系统 功能 模块 划分 ,将 数据 库 的 多 辑 结构 设 计 为 MySQL 5.5 x 


持 的 实际 数据 模型 ,如 表 12-2 ER 12-12 所 示 。 


表 12-2 用 户 管理 表 


T R 含 x 类 型 及 长 度 约 R 
user_id 用 户 ID int(11) EE nens 
primary key 
user name 用 户 名 char(50) 
user_password 密码 char(100) 
user_sex 性 别 char(2) 
user_birthday 生日 datetime 
user QQ QQ 号 码 int(11) 
user Email 电子 邮件 char(50) 
user_tel 电话 号 码 char(50) 
user_face 头像 char(100) 
user_sign 个 性 签名 Text 
user_grade 用 户 等 级 char(50) 
user_mark 用 户头 衔 int(11) 
user_topic 主题 数 int(11) 
user_wealth 财富 积分 int(11) 
user post 帖子 数 int(11) 
user group 门派 char(50) 
user_lastip 上 次 登录 IP char(15) 
user delnum 删除 数 int(11) 
user. friends 好 友 text 
user regtime 注册 时 间 datetime 
user_lasttime 最 近 登 录 时 间 datetime 
user locked 用 户 是 否 被 锁定 ENUM("false" , "true") default 'false' 
user_admin 用 户 是 否 为 后 台 管 理 员 ENUM("false","true") default 'false' 
user_password_a 密码 提示 问题 答案 char(100) 
user_password_q 密码 提示 问题 char(100) 
user_age 年 龄 int(11) 
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续 表 
字 段 含 x 类 型 及 长 度 约 * 
user secondname 姓氏 char(50) 
user_truename 真实 名 字 char(50) 
user_blood 血型 char(10) 
user shengxiao 生肖 char(10) 
user_nation 国家 char(50) 
user province 省 份 char(50) 
user_city 城市 char(50) 
表 12-3 ”用户 信箱 表 
T R *$ X 类 型 及 长 度 约 R 
not null 
msg_id 消息 ID int(11) auto increment 
primary key 
msg from 消息 来 源 char(50) default null 
msg_to 消息 目的 地 char(50) not null 
msg topic 消息 主题 char(100) not null 
msg_words 消息 内 容 mediumtext not null 
msg_time 消息 传送 时 间 datetime not null 
ll 
msg new 是 否 为 新 消息 enum( 'true', 'false') wa 
default 'true' 
msg belong 待 发 消息 enum( 'true', 'false') bot pall 
default 'true' 
e not null 
msg del 删除 消息 tinyint(1) default de 
ll 
msg del msgbin 接收 消息 enum('0','1') pies 
default '1" 
ll 
msg, del forebin 发 送 消息 enum('0','1') noren 
default '1' 
R 12-4 管理 员 表 
字 E * x 类 型 及 长 度 约 * 
admin_id 管理 员 ID int(11) 
primary key 
admin_name 管理 员 用 户 名 char(50) 
admin password 管理 员 密码 char(25) 
admin user 普通 用 户 名 char(50) 
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表 12-5 帮助 表 
字 段 含 x 类 型 及 长 度 约 * 
ca 
help. id 帮助 ID man NN 
primary key 
help name 帮助 主题 char(100) default null 
help_info 帮助 内 容 text 
R126 FAR 
F 段 含 x 类 型 及 长 度 约 * 
user_id 用 户 ID int(11) default null 
friend id 好 友 ID int(11) default null 
friend name 好 友 用 户 名 char(50) default null 
friend_addtime 添加 好 友 时 间 varchar(20) default null 
friend_Email 好 友 电 子 邮件 varchar(20) default null 
表 12-7 论坛 版 块 管理 表 
字 E £ x 类 型 及 长 度 约 R 
board. id 版 块 ID int(11) Kan DAN 
primary key 
board_isMother 是 否 为 主 版 块 ENUM ( 'true', 'false') default 'false' 
board bid 所 属 版 块 ID int(11) 
board name 版 块 名 称 char(50) 
board_info 版 块 信息 text 
board_master 版 主 char(50) 
board_img 版 块 图 标 char(100) 
board postnum 版 块 帖子 数 int(11) 
board_topicnum 版 块 主题 数 int(11) 
board_todaynum 版 块 今日 帖 数 int(11) 
board_lastreply 最 新 回复 帖 ID int(11) 
表 12-8 ”帖子 管理 表 
* E 含 x 类 型 及 长 度 约 * 
post, id 帖子 ID int) MEE 
primary key 
post boardid 帖子 所 属 版 块 int(11) 
post_user 发 帖 用 户 char(50) 
post_topicid 帖子 主题 ID int(11) 
post_replyid 帖子 回复 ID int(11) 
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续 表 
字 段 含 x 类 型 及 长 度 约 * 
post_content 帖子 内 容 text 
post_time 发 帖 时 间 datetime 
post_edittime 帖子 编辑 时 间 datetime 
post_ip 发 帖 IP char(15) 
表 12-9 论坛 基本 设置 表 
F 段 含 x 类 型 及 长 度 约 * 
i t 
setting kd 设置 ID int(11) R 
primary key 
setting_copyrights 版 权 设 置 text 
setting_author 设置 人 text 
论坛 是 否 处 于 开放 状 
setting_open 态 ,false 为 论坛 关闭 ，| ENUM('false', true) default 'true' 
true 为 开放 
setting_offwords 舍 人 字 设 置 text 
setting_badwords 污秽 字 设 置 text 
setting bbsname BBS 名 称 char(100) 
setting bbsurl BBS 访问 地 址 char(100) 
setting bbslogo BBS 图 标 char(100) 
setting info 信息 内 容 text 
setting maxonlinetime | 最 长 在 线 时 间 datetime 
setting_maxonline 最 大 在 线 数 int(11) 
setting_todaynum 今日 访问 量 int(11) 
setting_yesterdaynum 昨日 访问 量 int(11) 
setting_maxnum 最 大 访问 量 int(11) 
setting_totaltopic 总 主题 数 int(11) 
setting_totalpost 总 帖子 数 int(11) 
setting_totaluser 总 用 户 数 int(11) 
表 12-10 话题 管理 表 
字 段 含 x 类 型 及 长 度 约 * 
topic_id 话题 ID int(11) auto_increment primary key 
topic_boardid 话题 所 属 版 块 int(11) 
topic_user 发 布 话题 用 户 char(50) 
topic_name 话题 名 称 char(100) 


datetime 


topic_time 发 布 话题 时 间 
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续 表 
字 段 含 = 类 型 及 长 度 约 * 
topic_hits 话题 点 击 量 int(11) 
topic, replynum 话题 回复 数 int(11) 
topic, lastreplyid 最 新 回复 话题 ID int(11) 
topic top 是 否 为 置顶 帖 ENUM("false", "true") |default 'false' 
topic_best 是 否 为 精华 ENUM("false" ,"true") |default 'false' 
topic del 是 否 为 删除 帖 ENUM("false" ,"true") |default 'false' 
topic hot 是 否 为 热门 帖 ENUM("false", "true") |default 'false' 
表 12-11 友情 链接 表 
字 R *$& X 类 型 及 长 度 约 R 
to i t 
link id 链接 ID maD PM D 
primary key 
link name 链接 名 称 char(50) 
link_url 链接 地 址 char(100) 
link_info 链接 信息 text 
link_logo 链接 图 标 char(100) 
ll 
link islogo 链接 是 否 为 图 标 ENUMC 'true', 'false') piss 
default 'false' 
ll 
link pass 链接 过 审 ENUMI('true' false") pem 
default 'false' 
表 12-12 广告 表 
字 段 * x 类 型 及 长 度 约 * 
ad id 广告 ID int(11) not null 
ad_img 广告 图 片 char(100) null 
ad_url 广告 链接 地 址 char(100) null 
ad_title 广告 标题 char(100) null 


12.2.2 创建 数据 库 和 数据 表 


根据 数据 库 的 逻辑 结构 ,创建 数据 库 和 创建 数据 表 的 代码 如 下 : 


/ * 创建 数据 库 icefish* / 


drop database if exists icefish; 


CREATE DATABASE icefish DEFAULT CHARACTER SET gb2312; 


use icefish; 


/* 创建 用 户 管理 表 icefish user * / 


CREATE TABLE icefish_user ( 
user_id int(11) auto_increment PRIMARY KEY 
user_name char(50) , 
user_password char(100), 
user_sex char(2), 
user_birthda datetime, 
user_QQ int(11), 
user_Email char(50), 
user tel char( 50), 
user. face char( 100), 
user sign text, 
user grade char(50), 
user mark int(11), 
user topic int(11), 
user wealth int(11), 
user post int(11), 
user group char(50), 
user lastip char(15), 
user delnum int(11), 
user friends text, 
user regtime datetime, 
user lasttime datetime, 


user locked enum( "false", "true") default 'false', 
user admin enum( "false", "true") default 'false', 
user password a char(100), 
user password q char(100), 
user age int(11), 
user secondname char(50) , 
user truename char(50), 
user blood char(10), 
user shengxiao char(10), 
user nation char( 50), 
user province char(50) , 
user city char(50) 
) ENGINE = InnoDB DEFAULT CHARSET — gb2312; 


/ * 创建 用 户 信箱 表 icefish msg * / 
CREATE TABLE icefish msg ( 

msg id int(11) not null auto increment, 

msg from char(50) default null, 

msg to char(50) not null, 

msg topic char(100) not null, 

msg words mediumtext not null, 

msg time datetime not null, 


msg new enum( 'true', 'false') not null default 'true', 
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msg belong enum( 'true', 'false') not null default 'true', 
msg del tinyint(1) not null default '0', 
msg del msgbin enum( '0', '1') not null default '1', 
msg del forebin enum( '0', '1') not null default '1', 
PRIMARY KEY (msg id) 

) ENGINE= InnoDB DEFAULT CHARSET — gb2312; 


/* 创建 管理 员 表 icefish admin * / 
CREATE TABLE icefish_admin ( 
admin_id int(11) auto_increment PRIMARY KEY, 
admin_name char(50) , 
admin_password char(25) , 
admin_user char(50) 


ENGINE- InnoDB DEFAULT CHARSET — gb2312; 


/ * 创建 帮助 表 icefish_help * / 
CREATE TABLE icefish help ( 
help id int(11) auto increment PRIMARY KEY, 
help name char(100) default null, 
help info text 
) ENGINE= InnoDB DEFAULT CHARSET- gb2312; 


/* 创建 好 友 表 icefish friend * / 
CREATE TABLE icefish friend ( 
user id int(11) default null, 
friend id int(11) default null, 
friend name char(50) default null, 
friend addtime varchar(20) default null, 
friend email varchar(20) default null 


ENGINE- InnoDB DEFAULT CHARSET= gb2312; 


/ * 创建 论坛 版 块 管理 表 icefish board * / 
CREATE TABLE icefish board ( 
board id int(11) auto. increment PRIMARY KEY, 
board isMother enum( 'true', 'false') default 'false', /* 是 否 主 版 块 */ 
board_bid int(11) , 
board_name char(50) , 
board_info text , 
board master char(50), 
board img char(100), 
board postnum int(11), 
board topicnum int(11), 
board todaynum int(11), 
board lastreply int(11) / * 最 新 回复 帖 的 ID* / 
) ENGINE= InnoDB DEFAULT CHARSET-— gb2312; 
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/ * 创建 帖子 管理 表 icefish_post * / 
CREATE TABLE icefish_post ( 
post_id int(11) auto_increment PRIMARY KEY, 
post_boardid int(11) , 
post_user char(50) , 
post topicid int(11), 
post replyid int(11), 
post content text, 
post time datetime , 
post edittime datetime, 
post ip char(15) 
) ENGINE= InnoDB DEFAULT CHARSET — gb2312; 


/ * 创建 论坛 基本 设置 表 icefish setting * / 
CREATE TABLE icefish setting ( 
setting id int(11) auto increment PRIMARY KEY, 
setting copyrights text, 
setting author text, 
setting open enum( 'false', 'true') default 'true', 
/* 论坛 是 否 处 于 开放 状态 ,false 为 论坛 关闭 ,true 为 开放 * / 
setting_offwords text, 
setting_badwords text, 
setting bbsname char(100), 
setting bbsurl char(100), 
setting bbslogo char(100), 
setting info text, 
setting maxonlinetime datetime, 
setting maxonline int(11), 
setting todaynum int(11), 
setting yesterdaynum int(11), 
setting maxnum int(11), 
setting totaltopic int(11), 
setting totalpost int(11), 
setting totaluser int(11) 
) ENGINE= InnoDB DEFAULT CHARSET — gb2312; 


/ * 创建 话题 管理 表 icefish topic * / 
CREATE TABLE icefish topic € 
topic id int(11) auto increment PRIMARY KEY, 
topic boardid int(11) , 
topic user char(50) , 
topic name char(100) , 
topic time datetime , 
topic hits int(11), 
topic replynum int(11), 


222 网 络 编程 与 动态 网 站 制作 


) 


topic lastreplyid int(11), 

topic top enum("false", "true") default 'false', 
topic best enum("false", "true") default 'false', 
topic del enum( "false" , " true") default 'false', 

topic hot enum( "false", " true") default "false" 


ENGINE- InnoDB DEFAULT CHARSET — gb2312; 


/ * 创建 友情 链接 表 icefish link * / 
CREATE TABLE icefish link ( 


) 


link id int(11) auto. increment PRIMARY KEY, 

link name char(50) , 

link url char(100) , 

link info text, 

link logo char(100), 

link islogo enum( 'true', 'false') not null default "false", 
link pass enum( 'true', 'false') not null default 'false" 


ENGINE- InnoDB DEFAULT CHARSET — gb2312; 


/* 创 建 广告 表 icefish ad * / 
CREATE TABLE icefish ad ( 


ad id int(11) not null , 

ad img char(100) null, 

ad url char(100) null, 

ad title char(100) null 

ENGINE- InnoDB DEFAULT CHARSET — gb2312; 


12.3 系统 管理 


12.3.1 导入 相关 的 Jar 包 


在 本 章 中 新 建 一 个 Web 工程 icefish, 在 icefish 中 开发 本 系统 。 由 于 本 系统 采用 纯 Java 
数据 库 驱 动 程序 连接 MySQL 5. 5 数据 库 , 所 以 需要 把 Java 数据 库 驱 动 程序 mysql- 
connector-java-5. 0. 6. jar 复制 到 icefish/ WEB-INF/lib( 本 系统 虚拟 Web 服务 目录 ) 及 DIN 
Tomcat 6.0\common\lib(Web 服务 器 安装 目录 ) 中 。 需 要 注意 的 是 ,如 果 在 后 期 编码 调试 
过 程 中 出 现 “bad handshake error!1" 之 类 的 错误 提示 , 则 需要 把 其 他 版 本 的 Java 数据 库 驱动 


程序 删除 。 
12.3.2 JSP 页 面 管理 


本 论坛 系统 中 用 到 的 JSP 页 面 保存 在 icefish 工程 中 ,系统 中 有 些 JSP 页 面 使 用 到 css 
和 JavaScript. fi X CSS 与 JavaScript 的 知识 ,在 本 书 第 1 章 中 有 简单 介绍 ,如 果 想 深入 理 


解 和 掌握 这 两 方面 内 容 可 参考 查阅 相关 书籍 。 


/* 是 否 为 置顶 帖 * / 
/ * 是否 为 精华 帖 * / 
/* 是 否 为 删除 帖 * / 
/ * 是 否 为 热门 帖 */ 
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BBS 论坛 系统 首页 为 index. jsp, 后 续 操作 都 以 此 为 基础 展开 ,在 浏览 器 地 址 栏 中 输入 
http://localhost:8080/icefish 即 可 进入 首页 ,运行 效果 如 图 12-2 所 示 。 
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12-2 ”BBS 论坛 系统 首页 效果 


例子 代码 index. jsp 如 下 : 


<% @ page pageEncoding= "gb2312" % > 

<% @ page contentType= "text/html; charset= gb2312" %> 

<% (8 page language= "java" Mo > 

<% @page import= "java. sql. * "%> 

<% (page import= "java. util. * " 267» 

<% @include file= "top. jsp" ⁄> 

<% 

String sqll= "select * from icefish board WHERE board_isMother= 'true' order by board id asc"; 
ResultSet rs1 — null; 

rsl — indexBean. executeSQL (sgll) ; 

»— 

<% 

// 取 随 机 产生 的 认证 码 (4 位 数字 ) 

String sRand— 
// 生 成 随机 类 


Random random = new Random(); 
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for (int i=0;i<4;i 十 十 ){ 
String rand= String. valueOf( random. nextInt(10)) ; 
sRand-- — rand; 
) 
// 将 认证 码 存 人 SESSION 
session. setAttribute( "rand", sRand) ; 
%> 
<html> 
<head> 
<title> KMA [a] ?k f&ie Xs p WK fB i63z 8 38 # At 1< /vitle> 
<meta http-equiv— "Content-Type" content= "text/html; charset=gb2312"> 
<script Language= "Javascript" src— "include/form. js" 7 — /Script> 
</head> 
<body topmargin= "0" style= "text-align: center" > 
<table width="100%" height= "60" border="1" bordercolorlight= " # 7777ff" 
bordercolordark= " # 7777ff" style= "border-collapse : collapse" > 
<tr bgcolor=" # dff2ed"> 
<td valign= middle align=center> 
<%if(ad2_img= =null| |ad2_img. equals("")) (9677 
此 处 可 放置 广告 (首页 大 幅 广告 ) 
«X else( > 
<a href—"« % —ad2 url >"> 


<img border " onload— "javascript: if( this. width> 979) (this. width=979;)" 
src= "< Yi =ad2_img% >" title— "< Yg —ad2 title96 >"> 
</a> 
<%}%> 
</td> 
</tr> 
</table> 


<p> 
<table width= "100%" border= "1" bordercolorlight= " # 7777ff" bordercolordark= " # 7777ff" 
style= "border-collapse: collapse" > 
<tr> 
<td height="25" colspan="2" background= "images/skin/1/bg_td. gif"> 
</td> 
</tr> 
<tr bgcolor=" # dff2ed" valign="middle" align="center"> 
<td width= "50% "> 
<% 
request. setCharacterEncoding( "gb2312"); 
if(user login)( 
> 
<table height— "100%" valign="middle" align="center" border— "0"> 
<tr> 
<td width="30% "> 
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=!--img src="< % = userBean. getUser_Face()%>" 
onload= "javascript : if( this. widthœ>=120) {width=120; }"--> 
欢迎 您 登录 ! 
</td> 
<td> 
</td> 
</tr> 
</table> 
<% }else{%> 
<form method="post" name= "userlogin" action= "check. jsp" > <br> 
<table height="100%" valign="middle" align= "center" border= "0"> 
<tr> 
二 td 二 欢迎 访问 <b></b> ,您 还 没有 [ 
<a class 二 yh href= "reg.jsp">>šEJIf< /a>> ] 或 [ 
<a class= yh href= "login. jsp" ^ X 3E < /a7 ] 


</td> 
</tr> 
<tr> 
<> 4: 
< input type = "text" name =" user. name" size = " 13" > 
nbsp; &-nbsp; 
验证 码 : 
< input name= "checknumber" type="text" maxlength= "4" 
size="6"> 
< out. print(sRand) ; % > 
</td> 
</tr> 
xu 
二 td 二 密码 : 
<input type=" password" name= "user password" size="13"> 
&nbsp; 
Cookie: 
< select name= "save login" 
<option value— "no, time" selected ^ f f£ 
<option value— "one day" f £— X 
<option value— "one month" f ££ — H 
<option value— "one, year" ^ [i f£ — 4F 
</select> 
</td> 
</tr> 
<e> 
<td> 
<input type="submit" onclick="CheckLogin()" value=" % "> 
& nbsp; 


<input type— "reset" value— "3€ "> & nbsp; &nbsp; 
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[ <a class=yh href 一 "" 之 忘记 密码 <</a> | 
</td> 
</tr> 
</table> 
</form> 
<%)%> 
</td> 
<td width= "5026" i£ 1z 3E 4c fr UL BEBO — / td> 
</tr> 
<tr> 
<td bgcolor=" # dff2ed" colspan="2" height= "30"— 
</td> 
</tr> 
</table> <p> 
<% while(rsl.next())( 
String board_id= rsl.getString("board id"); 
n 
<table width "10076" border— "1" bordercolorlight— " # 7777FF" bordercolordark- " # 7777FF" 
style= "border-collapse: collapse" > 
<tr> 
<td height —"25" colspan= 


background = " images/skin/1/bg_td. gif" valign= 
"middle" > 
<b> 
<font color=" # ffffff"> &-nbsp; 
<img src= "images/skin/1/nofollow. gif"> 
<a class= xh href= "board. jsp?bm= — % = rs1. getString("board_name") Yo > 8- 
bmid=< % =rs1. getString(" board. id") 4 > &. 
board id— — % =rs1. getString("board id") % >"> 
<% — rs. getString(" board name") ⁄ > 
</a> 
</font> 
</td> 
</tr> 
<% 
String sql2= "select * from icefish board WHERE board_bid=" + board_id+" order by 
board_id asc" ; ResultSet rs2= null; 


= 


String board todaynum n 


=". 


String _board_topicnum ; 
String _board_postnum=""; 
rs2— indexBean. executeSQL (sql2) ; 
while(rs2. next()){ 
// 
board todaynum- rs2. getString(" board todaynum") ; 
if (. board. todaynum-— — null) ( 
board todaynum-— "0" ; 
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} 


// 

_board_topicnum= rs2. getString("board topicnum") ; 
if ( board topicnum- — null) { 

board topicnum- "0" ; 

) 

// 

board postnum- rs2. getString(" board postnum") ; 


if ( board postnui 一 nullD){ 


_board_postnum= 


85" valign="middle" align= "center" width= "43"> 


<img src= "images/skin/1/forum_nonews. gif" > 
</td> 
<td> 
<table width="100%" height="100%" border= "2" bordercolorlight= " # ffffff" 
bordercolordark — " & ffffff" style— " border-collapse: collapse" > 
<tr> 
<td valign="top"> 
<a class= zh href=" board. jsp?bm= < % = rs1. getString(" board 
name") ⁄ > &. 
bmid= — % =rs1. getString("board_id") %> &. 
board_id= — % =rs2. getString("board_id")% >"> 
<% =rs2. getString(" board name") 47» 
</a> 
<br> 
<img src= "images/skin/1/forum readme. gif" > 
<% — rs2. getString(" board. info") % > 


</td> 
<td width="27% "> 主题 : <br> R: <br> AW: </td> 
</tr> 
<tr bgcolor=" # dff2ed" height= "23"> 
<td> ME: 
<% =rs2. getString("board_master") ⁄ > 
</td> 
<td> 


<font color=" # ff0000"> 

<img src= "images/skin/1/forum today. gif" > 
<%= _board_todaynum% > 

</font><font color=" # 0000ff"> 

<img src= "images/skin/1/forum_topic. gif" > 
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< Ya = board topicnum 747» 
<img src= "images/skin/1/forum post. gif" > 
<%= board postnum 4 ></font></td> 
</tr> 
</table> 
</td> 
</tr> 
<%} 
rs2.close(); 
%> 
</table> <p> 
<%⁄) 
rsl.close(); 
%> 
<table width="100%" height= "100" border— "1" bordercolorlight=" # 7777ff" 
bordercolordark= " # 7777ff" style= "border-collapse: collapse" > 
<tr> 
<td height= "25" colspan— "2" background= "images/skin/1/bg_td. gif" > 
<b> 
<font color=" # ffffff" > 
=> 友情 论坛 ; 
<a class 一 xh href— "linkadd. jsp" > [ Hi Hi z fff && BE ] </a> 
</font> 
</td> 
</tr> 
<tr bgcolor=" # dff2ed" — — td valign= middle align= center> 
<table width="100%" border= "0"> 
<tr align=left> <td> 
<% 
String sgl3— "select * from icefish link where link_islogo= 'false' 
and link pass— 'true'"; 
ResultSet rs3— indexBean. executeSQL ( sql3) ; 
while(rs3. next) ) ( 
x» 
<a class—zh href= "< % — rs3. getString( "link url") %4 >"> 
<% — rs3. getString("link name") % ></a> 
<%) 
rs3.close(); 
%> 
</td></tr> 
<tr> <td><hr></hr> </td></tr> 
<tr align=left> <td> 
<% 
String sql4= "select * from icefish link where link_islogo= 'true' and link_ 
pass— 'true'"; 
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ResultSet rs4=indexBean. executeSQL(sql4) ; 
while(rs4. next()){ 
“> 


<a class 一 zh href=" <% — rs4. getString("link url") % >"> 


<img border= "0" width — "88" height — "31" src=" <% = rs4. getString 


("link logo") Ya >" 


title— "< Yi — rs4. getString( "link name") %>"> 


</a> 4 @ src 
x e cd 


rs3. close) ; 
indexBean. close) ; 
%> 
</td></tr> 
</table> 
</td></tr> 
</table> 
</body> 
</html> 


12.3.3 #445 Servlet 管理 


BBS 论坛 系统 中 使 用 的 组 件 与 Servlet 包 层 次 结构 如 
图 12-3 所 示 。 

1. bean 包 

bean 包 里 存放 的 Java 程序 是 实现 论坛 系统 主要 功能 所 
要 用 到 的 数据 模型 和 业务 模型 ,包括 用 户 数 据 、 帖 子 数据 、 版 
块 数据 、 友 情 链接 数据 以 及 数据 库 连 接 等 方面 。 

2. edit 包 

edit 包 里 存放 的 是 实现 论坛 系统 “在 线 编辑 文本 器 "功能 
相关 的 数据 模型 和 业务 模型 ,包括 设置 样式 .设置 全 屏幕 选 
项 .设置 内 容 选 项 等 。 

3. page 包 

page 包 里 存放 的 是 结果 集 处 理 相 关 的 业务 逻辑 ,包括 显 
示 总 页 数 、 总 记录 行 数 、 当 前 页 号 、 当 前 页 的 记录 条 数 、 分 页 大 
小 等 。 

4. servlet (9 

servlet 包 里 存放 的 是 论坛 系统 的 控制 器 。 具 体 控制 器 功 
能 将 在 12. 5 节 加 以 说 明 。 


4 @ bean 
国 AddnoticeBean java 
AdminBeanjava 
BoardBean java 
BoardDataBean java 
BoardViewBeanjava 
国 Connjava 
DelnoticeBeanjava 
HelpBeanjava 
HelpDataBean java 
|i) IndexBeanjava 
InsertmyfriendBean java 
LinkBean.java 
LinkDataBeanjava 
MymodifyBean java 
国 NoticeBean.java 
I) PostBeanjava 
Ij) PostDataBeanjava 
国 PostViewBeanjava 
RepasswordBean java 
国 UserBeanjava 
国 UserDataBean java 
4 © edit 
li) EditBeanjava 
B) EditWebhelper java 
E RemotePicjava 
国 Timestampjava 
li) UploadBeanjava 
加 UploadWebHelperjava 
4 @ page 
[Ë Pageablejava 
|) ResultSetPage java 
4 (B serviet 
Addnoticeservlet.java 
AddPostServlet java 
国 AddUserServletjava 
AdminLoginjava 
BoardServletjava 
CheckUser java 
Delnoticeservletjava 
HelpServletjava. 
国 InsertmyfriendServietjava 
LinkServletjava 
国 LoginServletjava 
国 ManageTopicjava 
国 MymodifyServletjava. 
RepasswordServletava 
国 ReplyPostServiet java. 


图 12-3 包 层次 结构 图 
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12.3.4 配置 文件 管理 


在 icefish 的 配置 文件 web. xml 中 ,部 署 了 BBS 论坛 系统 用 到 的 控制 器 。 
代码 模板 web. xml 如 下 : 


<?xml version="1.0" encoding="UTF-8"?> 
<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" 
"http://java. sun. com/dtd/web-app_2_3.dtd"> 
<web-app> 
<display-name> defaultroot</display-nameœ> 
<servlet> 
<servlet-name>œ>debugjsp</servlet-name> 
<description> 
Added to compile JSPs with debug info 
</description> 
<servlet-class>org. apache. jasper. servlet. JspServlet</servlet-class> 
<init-param> 
“< param-name> classdebuginfo< /param-name> 
<param-value> true</param-value> 
</init-param> 
<load-on-startup>3</load-on-startup> 
</servlet> 
<servlet-mapping> 
<servlet-name> debugjsp</servlet-name> 
<url-pattern> * .jsp</url-pattern> 
</servlet-mapping> 


<servlet> 

<servlet-name> AddPostServlet</servlet-name> 

servlet-class> net. icefish. servlet. AddPostServlet</servlet-class> 
</servlet> 
<servlet-mapping> 

<servlet-name> AddPostServlet< /servlet-name> 

<url-pattern>/ AddPostServlet— /url-pattern> 
</servlet-mapping> 


<servlet> 
<seryvlet-name> EditPostServlet</servlet-name> 
<servlet-class>net. icefish. servlet. EditPostServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<serylet-name> EditPostServlet</servlet-name> 
< url-pattern >/EditPostServlet</url-pattern> 
</servlet-mapping> 
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<servlet> 
<servlet-nameœ ReplyPostServlet< /servlet-name> 
<servlet-class> net. icefish. servlet. ReplyPostServlet— /servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name> ReplyPostServlet< /servlet-name> 
url-pattern> /ReplyPostServlet</url-pattern> 
</servlet-mapping> 


<serylet> 
—servlet-name>> AddUserServlet</servlet-name> 
servlet-class> net. icefish. servlet. AddUserServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name> AddUserServlet</servlet-name> 
url-pattern>/AddUserServlet< /url-pattern> 
</servlet-mapping> 


<servlet> 
<servlet-name> LoginServlet< /servlet-name> 
<servlet-class>net. icefish. servlet. LoginServlet< /servlet-class 
</servlet> 
<servlet-mapping> 
<servlet-name> LoginServlet</servlet-name> 
<url-pattern>/LoginServlet</url-pattern> 
</servlet-mapping> 


<servlet> 
<servlet-name>œ>CheckUser</servlet-name> 
servlet-class > net. icefish. servlet. CheckUser </servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-nameœ>CheckUser</servlet-name> 
<url-pattern>/CheckUser</url-pattern> 
</servlet-mapping> 


<servlet> 
<servlet-name> AdminLogin< /servlet-name> 
—servlet-class7 net. icefish. servlet. AdminLogin< /servlet-class 
</servlet> 
<servlet-mapping> 
<servlet-name> AdminLogin< /servlet-name> 
url-pattern> /admin/ AdminLogin< /url-pattern> 
< /servlet-mapping> 
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<servlet> 
—servlet-name> BoardServlet< /servlet-name2> 
servlet-class> net. icefish. servlet. BoardServlet< /servlet-class> 
</servlet> 
<servlet-mapping> 
—servlet-name> BoardServlet< /servlet-name> 
<url-pattern>/admin/ BoardServlet— /url-pattern> 
</servlet-mapping> 


<serylet> 

—servlet-name> HelpServlet</servlet-name> 

—servlet-class7 net. icefish. servlet. HelpServlet</servlet-class> 
</servlet> 
<servlet-mapping> 

<servlet-name> HelpServlet</servlet-name> 

url-pattern> /admin/ HelpServlet< /url-pattern> 
/servlet-mapping> 


<servlet> 
“<servlet-name> ADServlet</servlet-name> 
<servlet-class>net. icefish. servlet. ADServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name> ADServlet< /servlet-name> 
<url-pattern>/admin/ ADServlet< / url-pattern 
< /servlet-mapping— 


<servlet> 

<servlet-name> LinkServlet</servlet-name> 

—servlet-class net. icefish. servlet. LinkServlet</servlet-class> 
</servlet> 
<servlet-mapping> 

<servlet-name> LinkServlet</servlet-name> 

url-pattern> /admin/LinkServlet< /url-pattern> 
</servlet-mapping> 


<servlet> 
<servlet-name> Addnoticeservlet< /servlet-name> 
<Sservlet-class>>net.icefish. servlet. Addnoticeservlet< /servlet-class> 
</servlet> 
<servlet-mapping> 
<serylet-name> Addnoticeservlet< / servlet-name> 
url-pattern>/Addnoticeservlet< /url-pattern> 
< /servlet-mapping 
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<servlet> 
—servlet-name> Delnoticeservlet</servlet-name> 
<servlet-class>net. icefish. servlet. Delnoticeservlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<serylet-name> Delnoticeservlet</servlet-name> 
< url-pattern > /Delnoticeservlet</url-pattern> 
</servlet-mapping> 


<servlet> 

<servlet-name> MymodifyServlet< /servlet-name> 

<servlet-class> net. icefish. servlet. MymodifyServlet< /servlet-class> 
</servlet> 
<servlet-mapping> 

<servlet-name> MymodifyServlet< /servlet-name> 

url-pattern> / MymodifyServlet— /url-pattern> 
</servlet-mapping> 


<servlet> 
<servlet-name> InsertmyfriendServlet</servlet-name> 
<servlet-class>net. icefish. servlet. InsertmyfriendServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name> InsertmyfriendServlet</servlet-name> 
<url-pattern>/InsertmyfriendServlet</url-pattern> 
</servlet-mapping> 


<servlet> 
<servlet-nameœ>RepasswordServlet</servlet-name> 
<servlet-classœ>net. icefish. servlet. RepasswordServlet</servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-nameœ RepasswordServlet</servlet-name> 
<url-pattern>/RepasswordServlet</url-pattern> 
</servlet-mapping> 


<servlet> 
<servlet-name> ManageTopic< /servlet-name> 
—Sservlet-class7 net. icefish. servlet. ManageTopic< /servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-nameœManageTopic</servlet-name> 
< url-pattern > /managetopic< /url-pattern> 
</servlet-mapping> 


</web-app> 
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12.4 组 件 设 计 


本 系统 中 用 到 的 组 件 有 数据 库 连 接 与 关闭 、 实 体 模型 (数据 封装 bean) 和 业务 模型 ( 业 
务 处 理 bean) 。 


12.4.1 


数据 库 连 接 与 关闭 


数据 库 的 连接 与 关闭 是 由 bean 包 中 的 Conn 类 实现 的 ,类 中 主要 有 两 个 方法 : public 
static Connection connection ) 方 法 负责 连接 ,public void close() 方 法 负责 关闭 。 
代码 模板 Conn. java 如 下 : 


package net. icefish. bean; 


import java.sql. * ; 


public class Conn( 


gb2312"; 


) 


public static Connection connection() ( 
Connection conn— null; 
try{ 
Class. forName( "com. mysql. jdbc. Driver") . newInstance() ; 
) 
catch( Exception e) | 
System. out. print(e. toString() ) ; 


) 


try( 
String url— "jdbc:mysql://127.0.0.1:3306/icefish?useUnicode= true&-characterEncoding — 


// 数 据 库 名 统一 为 icefish, 表 名 与 字段 名 参照 系统 分 析 与 设计 
String user= "root"; // 设 置 用 户 名 
String password= "admin"; // 设 置 密码 
conn= DriverManager. getConnection( url, user, password) ; 
) 
catch( SQLException e) | 

System. out. print(e. toString()) ; 
) 


return conn; 


public void close) í 


try( 
conn. close) ; 
) 
catch( SQLException e) | 
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System. out. print(e. toString()); 


12.4.2 实体 模型 


实体 模型 主要 用 于 封装 JSP 页 面 提交 的 信息 以 及 与 数据 库 交 互 的 数据 传递 。 本 系统 
共 使 用 了 七 个 实体 模型 : 管理 员 (AdminBean)、 用 户 (UserBean)、 版 块 (BoardBean)、 帖 子 
(PostBean) ,公告 (NoticeBean) ,帮助 (HelpBean) 和 友情 链接 (LinkBean)。 

代码 模板 AdminBean. java 如 下 : 


package net. icefish. bean; 


public class AdminBean ( 
private String admin_name; 
private String admin_password; 
private String admin_user; 
private boolean admin_login; 


public AdminBean() { 
admin name- null; 
admin password— null; 
admin, user— null; 


admin login- false; 


public String getAdmin Name()( 
return admin name; 

) 

public void setAdmin Name(String admin name)( 
this. admin name- admin name; 


public String getAdmin Password() ( 
return admin password; 

) 

public void setAdmin Password(String admin password) ( 
this. admin password— admin password; 


) 


public String getAdmin_User() í 
return admin user; 
) 


public void setAdmin User(String admin user) | 
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this. admin_user=admin_user; 


public boolean get Admin_Login() í 
return admin_login; 
) 
public void setAdmin Login(boolean admin login) ( 
this. admin login admin login; 
) 
) 


代码 模板 UserBean. java 如 下 : 


package net. icefish. bean; 

public class UserBean ( 
private int user id; 
private String user name; 
private String user password; 
private String user password q; 
private String user password a; 
private String user sex; 
private String user age; 
private String user 009; 
private String user birthday; 
private String user Email; 
private String user tel; 
private String user face; 
private String user sign; 
private String user grade; 
private String user mark; 
private String user topic; 
private String user. wealth; 
private String user group; 
private String user post; 
private String user lastip; 
private String user delnum; 
private String user friends; 
private String user regtime; 
private String user lasttime; 
private boolean user admin; 
private boolean user locked; 
private boolean user login; 


public UserBean() í 
user mark— "0"; 
0"; 


user topic 
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user_post="0"; 


user_wealth 


Q^ ; 


user login false; 


user delnum— 


) 

public int getUser. IDO { 
return user id; 

) 

public void setUser ID(int user_id) { 
this. user id— user id; 

) 

public String getUser Name( | 
return user name; 

) 

public void setUser Name(String user name) 
this. user name— user name; 

) 

public String getUser. Password() | 
return user password; 

) 

public void setUser. Password(String user password) ( 
this. user password- user password; 

) 

public String getUser Password a()í 
return user password a; 

) 

public void setUser Password a(String user password a) 
this. user password a-user password a; 

) 

public String getUser Password q(1 
return user password q; 

) 

public void setUser Password q(String user password q)í 
this.user password q- user password q; 

) 

public String getUser Age(( 
return user age; 

) 

public void setUser Age(String user age)í 
this. user age— user age; 

) 

public String getUser SexO | 
return user sex; 

) 


public void setUser Sex(String user sex) 
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this.user_sex=user_sex; 

) 

public String getUser QQO( 
return user QQ; 

) 

public void setUser QQ(String user QQ){ 
this. user QQ— user QQ; 

) 

public String getUser Birthday í 
return user birthday; 

) 

public void setUser Birthday(String user birthday) ( 
this. user birthday- user birthday; 

) 

public String getUser Email() | 
return user Email; 

) 

public void setUser Email(String user Email) í 
this. user Email— user. Email ; 

) 

public String getUser Face | 
return user face; 

) 

public void setUser Face(String user face)í 
this. user face— user face; 

) 

public String getUser. Tel() | 
return user tel; 

) 

public void setUser Tel(String user tel) | 
this. user tel— user tel; 

) 

public String getUser Sign( | 
return user sign; 

) 

public void setUser Sign(String user sign)í 
this. user sign- user sign; 

) 

public String getUser Grade() | 
return user grade; 

) 

public void setUser. Grade(String user grade)( 
this. user grade— user grade; 

) 

public String getUser TopicO í 
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return user_topic; 

} 

public void setUser_Topic(String user_topic) | 
this. user topic— user topic; 

) 

public String getUser Wealth()1 
return user wealth; 

) 

public void setUser Wealth(String user wealth) í 
this. user wealth- user wealth; 

) 

public String getUser Mark()( 
return user mark; 

) 

public void setUser Mark(String user mark) ( 
this. user mark— user. mark; 

) 

public String getUser GroupO | 
return user group; 

) 

public void setUser Group(String user group)( 
this. user group-— user group; 

) 

public String getUser. Post() í 
return user post; 

) 

public void setUser. Post(String user post)( 
this. user post— user post; 

) 

public String getUser LastIPO í 
return user lastip; 

) 

public void setUser  LastIP(String user lastip) ( 
this. user lastip— user lastip; 

) 

public String getUser Delnum() ( 
return user delnum; 

) 

public void setUser Delnum(String user delnum) ( 
this. user delnum— user. delnum; 

) 

public String getUser Friends() | 
return user friends; 

) 


public void setUser Friends(String user friends) ( 
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this. user_friends 一 user_friends; 
) 
public String getUser. Regtime() ( 
return user regtime; 
) 
public void setUser Regtime(String user regtime) | 
this. user regtime— user regtime; 
) 
public String getUser. Lasttime() | 
return user lasttime; 
) 
public void setUser  Lasttime(String user lasttime) ( 
this. user lasttime- user lasttime; 
) 
public boolean getUser. Admin() í 
return user admin; 
) 
public void setUser Admin(boolean user admin) | 
this. user admin— user admin; 
) 
public boolean getUser. Locked() ( 
return user locked; 
) 
public void setUser Locked(boolean user locked)( 
this. user locked— user locked; 
) 
public boolean getUser Login()( 
return user login; 
) 
public void setUser. Login( boolean user login) | 
this. user login— user login; 
) 
) 


代码 模板 BoardBean. java 如 下 : 


public class BoardBean ( 
private String board_name; 
private String board_info; 
private String board_id; 
private String board_bid; 
private String board_master; 
private boolean board_isMother; 
private String board_postnum; 
private String board_topicnum; 
private String board_lastreply; 


第 12 章 ”动态 网 站 开发 综合 实例 241 


private String board_todaynum; 
private String board_img; 


public BoardBean() { 
board_name= null; 
board_info= null; 
board_master= null; 
board_bid=null; 
board_isMother= true; 

) 

public String getBoard IDO { 
return board id; 

) 

public void setBoard ID(String board id)( 
this. board. id— board id; 

) 

public String getBoard Name() { 
return board name; 

) 

public void setBoard Name(String board name) | 
this. board name- board name; 

) 

public String getBoard Info() í 
return board info; 

) 

public void setBoard Info(String board info) | 
this. board. info— board info; 

) 

public String getBoard. BID() ( 
return board bid; 

) 

public void setBoard. BID(String board bid) | 
this. board bid— board bid; 

) 

public String getBoard Master() í 
return board master; 

) 

public void setBoard Master(String board master) | 
this. board master— board master; 

) 

public boolean getBoard IsMother( ) { 
return board isMother; 

) 

public void setBoard IsMother( boolean board isMother) í 
this. board isMother— board isMother; 
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} 

public String getBoard_Postnum() í 
return board_postnum; 

) 

public void setBoard Postnum(String board postnum) í 
this. board. postnum- board postnum; 

) 

public String getBoard Topicnum() | 
return board topicnum ; 

) 

public void setBoard Topicnum(String board topicnum) í 
this. board topicnum- board topicnum; 

) 

public String getBoard Todaynum() | 
return board todaynum; 

) 

public void setBoard Todaynum(String board todaynum) í 
this. board todaynum- board todaynum; 

) 

public String getBoard  Lastreply() ( 
return board lastreply; 

) 

public void setBoard Lastreply(String board lastreply) í 
this. board. lastreply— board lastreply ; 

) 

public String getBoard Img()( 
return board img; 

) 

public void setBoard Img(String board img) 
this. board img board img; 

) 

) 


代码 模板 PostBean. java 如 下 : 


package net. icefish. bean; 

public class PostBean { 
private String board id; 
private String topic id; 
private String post content; 
private String topic name; 
private String user name; 


public PostBean() í 
) 
public String getBoard Id) ( 
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return board id; 

) 

public void setBoard Id(String board id) 
this. board id— board id; 

) 

public String getTopic Id()( 
return topic id; 

) 

public void setTopic Id(String topic id) í 
this. topic id- topic id; 

) 

public String getPost Content() í 
return post content; 

) 

public void setPost Content(String post content) í 
this. post content— post content; 

) 

public String getTopic Name() | 
return topic name; 

) 

public void setTopic Name(String topic name) | 
this.topic name- topic name; 

) 

public String getUser Name()( 
return user name; 

) 

public void setUser Name(String user name) 
this. user name- user name; 

) 

) 


代码 模板 NoticeBean. java 如 下 : 


public class NoticeBean ( 
private String notice_id; 
private String notice author; 
private String board name; 
private String notice music; 
private String notice title; 
private String notice date; 
private String notice content; 
private String xg; 

public NoticeBean( ) 

{ 

) 


public void setNotice id(String notice id) 


244 网 络 编程 与 动态 网 站 制作 


this. notice_id = notice_id; 
) 
public void setCheckbox(String xg) 
{ 
this. xg = xg; 
) 
public void setNotice Author(String notice author) 
{ 
this. notice_author = notice_author; 
) 
public void setBoard_Name(String board_name) 
{ 
this. board. name = board name; 
) 
public void setNotice Music(String notice music) 
{ 


this. notice_music = notice_music; 


public void setNotice_Title(String notice_title) 
{ 
this.notice title — notice title; 
) 
public void setNotice_Date(String notice_date) 
( 
this. notice date — notice date; 
) 
public void setNotice Content(String notice content) 
{ 
this.notice content — notice content; 
) 
public String getNotice IDO 
{ 
return notice_id; 
) 
public String getNotice_Author() 
{ 
return notice author; 
) 
public String getBoard Name() 
( 
return board name; 
) 
public String getNotice Music() 
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return notice_music; 
} 
public String getNotice_Title() 
( 
return notice title; 
) 
public String getNotice Date() 
{ 
return notice_date; 
) 
public String getNotice Content() 
{ 
return notice_content; 
) 
public String getCheckbox() 
( 
return xg; 
) 
) 


代码 模板 HelpBean. java 如 下 : 


public class HelpBean í 
private String help_id; 
private String help_name; 
private String help_info; 


public HelpBean() í 
help_name 一 null; 
help_info= null; 
) 
public String getHelp IDO( 
return help id; 
) 
public void setHelp ID(String help id) 
this. help id—help id; 
) 
public String getHelp Name()1 
return help name; 
) 
public void setHelp Name(String help name) í 
this. help name— help name; 
) 
public String getHelp InfoO | 
return help info; 
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) 
public void setHelp Info(String help. info) ( 
this. help info—help info; 
) 
) 


代码 模板 LinkBean. java 如 下 : 


public class LinkBean ( 
private String link_name; 
private String link_url; 
private String link_id; 
private boolean link islogo; 
private boolean link pass; 
private String link info; 
private String link logo; 


public LinkBean() ( 
link pass- false; 
link islogo false; 

) 

public String getLink IDO | 
return link id; 

) 

public void setLink ID(String link id)( 
this.link id—link id; 

) 

public String getLink Name()( 
return link name; 

) 

public void setLink Name(String link name) ( 
this.link name- link name; 

) 

public String getLink Info()( 
return link info; 

) 

public void setLink Info(String link info)( 
this.link info—link info; 

) 

public String getLink Logo()( 
return link logo; 

) 

public void setLink Logo(String link logo) | 
this. link logo link logo; 

) 

public String getLink Url()( 
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return link_url; 

} 

public void setLink Url(String link urb ( 
this.link url—link url; 

) 

public boolean getLink Pass()( 
return link pass; 

) 

public void setLink Pass(boolean link pass)í 
this.link pass-—link pass; 

) 

public boolean getLink Islogo()( 
return link islogo; 

) 

public void setLink Islogo(boolean link islogo) ( 
this.link islogo— link islogo; 

) 

) 


12.4.3 业务 模型 


本 系统 中 主要 用 到 的 业务 模型 有 如 下 几 个 : UserDataBean、BoardDataBean、 


BoardViewBean, PostDataBean, PostViewBean, AddnoticeBean, DelnoticeBean, HelpDataBean, 
InsertmyfriendBean , MymodifyBean , RepasswordBean , LinkDataBean , 


UserDataBean 业务 模型 的 主要 功能 是 : 新 用 户 注册 并 在 注册 时 验证 用 户 是 否 已 存在 ; 


用 户 登录 并 在 登录 时 验证 用 户 名 与 密码 是 否 能 通过 。 


代码 模板 UserDataBean. java 如 下 : 


public class UserDataBean ( 

private Connection conn; 

public UserDataBean() í 
this. conn Conn. connection( ) ; 

) 

// 注 册 时 验证 用 户 是 否 已 存在 

public boolean checkUser( UserBean userBean) | 
boolean flag = false; 
String user name- userBean. getUser. Name() ; 
Statement stmt— null; 
try{ 

stmt= conn. createStatement() ; 


ResultSet rs 一 stmt.executeQuery("select * from icefish_user where 


user name— '" +user_name+"'"); 
if(!rs. next) ( 
flag= true; 
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)else( 
flag false; 
) 
rs. close ; 
stmt. close() ; 
conn. close) ; 
) 
catch( SQLException e) í 
flag false; 
System. out. println(e. toString()) ; 
) 
return flag; 
) 
// 登 录 时 验证 用 户 名 与 密码 是 否 能 通过 
public boolean loginUser( UserBean userBean) ( 
boolean flag- false; 
String user name— userBean. getUser Name() ; 
String user password userBean. getUser. Password() ; 
Statement stmt— null; 
try{ 
stmt— conn. createStatement() ; 
ResultSet rs— stmt. executeQuery ("select * from icefish user where user name— '" + user | 
name--"' and user password '"-- user password-d-" '"); 
ifC! rs. next) ( 
flag— false; 
)else( 
while(rs. next()) | 
String QQ= rs. getString(" user QQ"); 
userBean. setUser. Admin(rs. getBoolean("user admin")) ; 
userBean. setUser Age(rs. getString("user age")); 
userBean. setUser  Birthday(rs. getString("user birthday")); 
userBean. setUser Delnum(rs. getString("user delnum")) ; 
userBean.setUser Email(rs.getString("user Email")) ; 
userBean.setUser Face(rs. getString("user face")); 
userBean. setUser  Friends(rs. getString("user friends")) ; 
userBean. setUser  Grade(rs. getString("user grade")); 
userBean. setUser Group(rs. getString("user group")) ; 
userBean. setUser ID(rs. getInt("user id")); 
userBean. setUser. LastIP(rs. getString("user lastip")); 
userBean. setUser. Lasttime(rs. getString("user lasttime")) ; 
userBean. setUser. Locked(rs. getBoolean("user locked")); 
userBean. setUser Mark(rs. getString("user mark")); 
userBean.setUser Password a(rs.getString("user password a")); 
userBean.setUser Password q(rs.getString("user password q")); 
userBean. setUser Post(rs. getString("user post")); 
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userBean. setUser QQ( QQ); 

userBean. setUser_Regtime(rs. getString("user regtime")) ; 
userBean. setUser. Sex(rs. getString("user sex")); 
userBean. setUser Sign(rs. getString("user sign")); 
userBean. setUser Tel(rs. getString("user. tel")); 
userBean. setUser. Topic(rs. getString("user topic")); 
userBean. setUser_ Wealth(rs. getString("user wealth")); 


) 
flag= true; 
) 
rs.close() ; 
stmt. close() ; 
conn. close() ; 
) 
catch( SQLException e) | 
flag false; 
System. out. println(e. toString()) ; 


return flag; 


) 
// 后 台 管 理 登录 验证 
public boolean adminLogin( AdminBean adminBean) | 
boolean flag= false; 
String admin name-— adminBean. getAdmin Name() ; 
String admin password— adminBean. getAdmin Password() ; 
String admin user— adminBean. getAdmin User() ; 
Statement stmt— null; 
try( 
stmt— conn. createStatement() ; 


ResultSet rs— stmt. executeQuery("select * from icefish admin where admin name 
admin named-"' and admin password-— '"--admin password--"' and admin user— '" +-admin_user+ 
if( !rs. next) ( 
flag false; 
Jelse( 
flag true; 
) 
rs. close(); 
stmt. close() ; 
conn. close) ; 
) 
catch( SQLException e) í 
flag= false; 
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System. out. println(e. toString()) ; 


return flag; 

) 

// 新 用 户 注册 

public boolean addUser( UserBean userBean) | 
boolean flag= false; 


PreparedStatement pstmtl = null; 
try{ 
pstmt1 = conn. prepareStatement( "insert into icefish user( 
user name, user password,user password q,user password a,user sex,user Email, user mark, user topic, 


user wealth, user post, user delnum, user regtime) values(?, ?, ?, ?, ?,?, ?, ?, ?,?,?, now())"); 
pstmtl.setString(1, userBean. getUser Name()) ; 
pstmtl.setString(2, userBean. getUser Password()) ; 
pstmtl.setString(3, userBean. getUser Password q()); 
pstmtl.setString(4, userBean. getUser Password a()); 
pstmtl.setString(5, userBean. getUser Sex()); 
pstmtl.setString(6, userBean. getUser Email()) ; 
pstmtl.setInt(7, 0); 
pstmtl.setInt(8, 0); 
pstmtl.setInt(9, 0); 
pstmtl.setInt(10, 0); 
pstmtl.setInt(11, 0); 
int result] = pstmtl.executeUpdate() ; 
if (resultl > 0)( 
flag — true; 
) 
else( 
flag = false; 

) 
pstmtl.close() ; 
conn.close() ; 

) 

catch( SQLException e) | 
flag= false; 
System. out. println(e. toString()) ; 

) 

return flag; 

) 
) 


BoardDataBean 业务 模型 的 主要 功能 是 : 增加 版 块 、 修 改版 块 和 删除 版 块 。 
代码 模板 BoardDataBean. java 如 下 : 
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public class BoardDataBean { 
private Connection conn; 


public BoardDataBean() | 
this. conn— Conn. connection() ; 

) 

// 增 加 新 版 块 

public boolean addBoard( BoardBean boardBean) | 
boolean flag= false; 


boolean board isMother- boardBean. getBoard IsMother() ; 
if(board isMother) ( 
PreparedStatement pstmtl = null; 
try{ 
pstmt1= conn. prepareStatement("insert into icefish board 
(board name, board info, board isMother, board master, board postnum, board topicnum, board _ 
todaynum) values(?, ?, 'true', ?,0,0,0)"); 
pstmtl.setString(1, boardBean. getBoard Name()); 
pstmtl.setString(2, boardBean. getBoard Info()) ; 
pstmtl.setString(3, boardBean. getBoard Master()) ; 
int result] = pstmtl . executeUpdate() ; 
if (result > 0)( 
flag — true; 
} 
else{ 
flag = false; 
} 
pstmtl.close() ; 
conn. close() ; 
) 
catch( SQLException e) í 
flag false; 
System. out. println(e. toString( )) ; 
) 
)else( 
PreparedStatement pstmtl = null; 
try{ 
pstmtl — conn. prepareStatement( "insert into icefish board 
(board name, board info, board isMother, board master, board bid, board postnum, board topicnum, 
board todaynum) values(?, ?, '£alse', ?,?,0,0,0)"); 
pstmtl.setString(1, boardBean.getBoard Name()); 
pstmtl.setString(2, boardBean. getBoard Info()) ; 
pstmtl.setString(3, boardBean. getBoard Master()) ; 
pstmtl.setString(4, boardBean.getBoard BID()) ; 
int result] = pstmtl. executeUpdate() ; 
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if (resultl — 0) ( 
flag = true; 
) 
else{ 
flag = false; 
) 
pstmtl.close() ; 
conn. close() ; 
) 
catch( SQLException e) | 
flag= false; 
System. out. println(e. toString()) ; 


return flag; 
) 
// 修 改版 块 信息 
public boolean editBoard( BoardBean boardBean) | 
boolean flag= false; 
boolean board isMother— boardBean. getBoard IsMother() ; 
if(board isMother) { 
PreparedStatement pstmt1 = null; 
tryl 
pstmtl = conn. prepareStatement ("update icefish board set board name-—?, board | 
info—?, board isMother— 'true', board master—?, board postnum- 0, board topicnum- 0, board 
todaynum=0 where board id— "--boardBean. getBoard ID()) ; 
pstmtl.setString(1, boardBean. getBoard Name()) ; 
pstmtl.setString(2, boardBean.getBoard Info()) ; 
pstmtl.setString(3, boardBean.getBoard Master()); 
int result] = pstmtl.executeUpdate() ; 
if (resultl > 0){ 
flag — true; 
) 
else( 
flag — false; 
) 
pstmtl.close() ; 
conn. close() ; 
) 
catch( SQLException e) í 
flag— false; 
System. out. println(e. toString()) ; 


Jelse( 
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PreparedStatement pstmt1=null; 
try{ 


pstmtl = conn. prepareStatement ("update icefish board set board_name=?, board_ 


? 


info=?, board_isMother = 'false', board_master=?, board bid 
十 boardBean. getBoard IDO) ; 
pstmtl.setString(1, boardBean. getBoard Name()); 
pstmtl.setString(2, boardBean. getBoard Info()) ; 
pstmtl.setString(3, boardBean. getBoard Master()) ; 
pstmtl.setString(4, boardBean.getBoard BID()); 

int result] = pstmtl. executeUpdate() ; 

if (resultl > 0)( 


flag — true; 


topicnum—0, board todaynum- 0 where board id 


) 
else{ 
flag = false; 
) 
pstmtl.close() ; 
conn. close() ; 
) 
catch( SQLException e) | 
flag— false; 
System. out. println(e. toString() ) ; 


) 
return flag; 
) 
// 删 除 版 块 
public boolean delBoard(BoardBean boardBean) ( 
boolean flag= false; 
boolean board_isMother= boardBean. getBoard IsMother( ) ; 
if(board isMother) ( 
// 如 果 所 删除 的 版 块 为 论坛 一 级 分 类 版 块 , 则 连同 其 下 属 的 子 版 块 也 删除 ,包括 所 有 帖子 
PreparedStatement pstmt1 一 null; 
PreparedStatement pstmt2 一 null; 
PreparedStatement pstmt3 一 null; 
PreparedStatement pstmt4 一 null; 
try( 
pstmt4— conn. prepareStatement("select * from icefish board where board bid—?"); 
pstmt4. setString(1, boardBean. getBoard IDO); 
ResultSet rs4— pstmt4. executeQuery() ; 
while(rs4. next()) { 
PreparedStatement pstmt5 — null; 
pstmt5 = conn. prepareStatement( "delete from icefish topic where topic boardid— ?") ; 
pstmt5. setString(1, rs4. getString("board id")); 
int result5— pstmt5. executeUpdate() ; 


board postnum = 0, board | 
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PreparedStatement pstmt6 — null; 
pstmt6— conn. prepareStatement( "delete from icefish post where post. boardid— ?") ; 
pstmt6. setString(1l, rs4. getString(" board id")); 
int result6 = pstmt6. executeUpdate() ; 
if (result6>0 && result5>0){ 
flag = true; 
) 
else( 
flag — false; 
) 
pstmt5. closeO ; 
pstmt6. close() ; 
) 
pstmt4. close() ; 
pstmtl = conn. prepareStatement ( " delete from icefish_ board where board id —? or 
board bid—?"); 
pstmtl.setString(1, boardBean. getBoard IDO); 
pstmtl.setString(2, boardBean. getBoard IDO); 
int result] = pstmtl. executeUpdate() ; 
if (resultl > 0)( 
flag — true; 
) 
elset 
flag — false; 
) 
pstmtl.close() ; 
pstmt2— conn. prepareStatement("delete from icefish topic where topic boardid— ?") ; 
pstmt2. setString(1, boardBean.getBoard IDO); 
int result2— pstmt2. executeUpdate() ; 
pstmt3— conn. prepareStatement( "delete from icefish post where post boardid— ?") ; 
pstmt3. setString(1, boardBean.getBoard IDO); 
int result3— pstmt3. executeUpdate() ; 
if (result3770 & & result2>0){ 
flag — true; 
) 
else( 
flag = false; 
) 
pstmt2. close() ; 
pstmt3. close() ; 
conn. close() ; 
) 
catch( SQLException e) ( 
flag false; 
System. out. println(e. toString()) ; 
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} 
}else{ 
PreparedStatement pstmt1 = null; 
PreparedStatement pstmt2= null; 
PreparedStatement pstmt3= null; 
try( 
pstmtl = conn. prepareStatement( "delete from icefish_board where board id— ?"); 
pstmtl.setString(1, boardBean. getBoard IDO); 
int result] = pstmtl . executeUpdate() ; 
if (resultl > 0)( 
flag — true; 
) 
else{ 
{flag = false; 
} 
pstmtl. close() ; 
pstmt2= conn. prepareStatement("delete from icefish_topic where topic_boardid=?") ; 
pstmt2. setString(1, boardBean. getBoard IDO); 
int result2= pstmt2. executeUpdate() ; 
if (result2 — 0)( 
flag — true; 
) 
else{ 
flag = false; 
} 
pstmt2. close() ; 
pstmt3= conn. prepareStatement( "delete from icefish post where post boardid— ?") ; 
pstmt3. setString(1, boardBean.getBoard IDO); 
int result3— pstmt3. executeUpdate() ; 
if (result3 — 0) ( 
flag = true; 
) 
else( 
flag = false; 
) 
pstmt3. close() ; 
conn. close() ; 
) 
catch( SQLException e) | 
flag— false; 
System. out. println(e. toString()) ; 
) 
) 


return flag; 
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BoardViewBean 业务 模型 的 主要 功能 是 : 浏览 版 块 。 
代码 模板 BoardViewBean. java 如 下 ; 


public class BoardViewBean { 
private Connection conn; 
private ResultSet rs; 


public Board ViewBean() ( 
this. conn— Conn. connection() ; 
) 
public ResultSet executeSQL (String sql) í 
try{ 
Statement stmt = conn. createStatement() ; // 语 句 接口 
rs = stmt. executeQuery(sql) ; // 获 得 结果 集 
) 
catch( SQLException e) í 
System. out. print(e. toString()) ; 
) 
//conn. close(); 
return rs; 
} 
public void close) ( 
try{ 
conn. close(); 
) 
catch( SQLException e) | 
System. out. print(e. toString()) ; 


) 
) 


PostDataBean 业务 模型 的 主要 功能 是 : 将 相关 信息 存 人 帖子 数据 表 和 话题 数据 表 。 
代码 模板 PostDataBean 如 下 : 


public class PostDataBean { 


private Connection conn; 


public PostDataBean( ) | 
this. conn= Conn. connection( ) ; 

) 

public boolean addPost(PostBean postBean) ( 
boolean flag- false; 
Statement stmt— null; 
PreparedStatement pstmt1 — null; 
PreparedStatement pstmt2— null; 
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try{ 
// 把 相关 信息 存 人 表 icefish_topic, 并 获得 当前 插入 的 topic 的 ID 
stmt= conn. createStatement() ; 
pstmtl — conn. prepareStatement( "insert into icefish topic 
(topic. boardid, topic name, topic user,topic time) values(?,?,?, now()) "5 ; 
pstmtl.setString(1, postBean. getBoard Id()) ; 
pstmtl.setString(2, postBean. getTopic Name()) ; 
pstmtl.setString(3, postBean. getUser Name()); 
int result] = pstmtl.executeUpdate() ; 
if (resultl > 0)( 
flag — true; 
) 
else( 
flag — false; 
) 
ResultSet rs= stmt. executeQuery( "select @ (2 identity as 'topic id'"); 
while(rs. next )) | 
String topic id— rs. getString("topic id"); 
// 把 相关 信息 存 人 表 icefish post 
pstmt2= conn. prepareStatement( "insert into icefish post 
(post. topicid, post. content, post. boardid, post user, post time) values(?,?,?, ?, now) "); 
pstmt2. setString(1, topic id); 
pstmt2. setString(2, postBean. getPost Content()) ; 
pstmt2. setString(3, postBean. getBoard Id()) ; 
pstmt2. setString(4, postBean. getUser Name()) ; 
int result2— pstmt2. executeUpdate() ; 
if (result2 > 0)( 
flag — true; 
) 
else( 
flag = false; 
) 
pstmt2. close() ; 
) 
rs. close); 
stmt. close) ; 
pstmtl.close() ; 
conn.close() ; 
) 
catch( SQLException e) | 
flag false; 
System. out. println(e. toString()) ; 
) 


return flag; 
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public boolean replyPost(PostBean postBean) ( 
boolean flag- false; 
PreparedStatement pstmtl = null; 
try{ 
String topic_id= postBean. getTopic Id() ; 
// 把 相关 信息 存 人 表 icefish post 
pstmt1= conn. prepareStatement( "insert into icefish post 
(post topicid, post content, post boardid, post user, post time) values(?,?, ?, ?, now()) "); 
pstmtl.setString(1l, topic id); 
pstmtl.setString(2, postBean. getPost Content()) ; 
pstmtl.setString(3, postBean. getBoard Id()); 
pstmtl.setString(4, postBean. getUser Name()) ; 
int result2— pstmtl . executeUpdate() ; 
if (result2 — 0)( 
flag — true; 
) 
else{ 
flag = false; 
) 
pstmtl.close() ; 
pstmtl.close() ; 
conn. close() ; 
) 
catch( SQLException e) | 
flag false; 
System. out. println(e. toString()) ; 
) 
return flag; 
) 
) 


PostViewBean 业务 模型 的 主要 功能 是 : 浏览 帖子 。 
代码 模板 PostViewBean. java 如 下 : 


public class PostViewBean ( 
private Connection conn; 
private ResultSet rs; 


public PostViewBeanO ( 
this. conn Conn. connection( ) ; 
) 
public ResultSet executeSQL (String sql) { 
try{ 
Statement stmt = conn. createStatement() ; // 语 句 接口 


rs = stmt. executeQuery(sql) ; // 获 得 结果 集 
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catch( SQLException e){ 
System. out. print(e. toString() ) ; 


) 
return rs; 
) 
public void close) ( 
try( 
conn. close() ; 
) 
catch( SQLException e) í 
System. out. print(e. toString()) ; 
) 
) 
) 


AddnoticeBean 业务 模型 的 主要 功能 是 : 增加 公告 。 
代码 模板 AddnoticeBean. java 如 下 : 


public class AddnoticeBean | 
private Connection conn; 


public AddnoticeBean( ) | 
this. conn— Conn. connection() ; 
) 
public void ggBe( NoticeBean a) ( 
PreparedStatement pstmt— null; 
try{ 
pstmt= conn. prepareStatement("insert into icefish notice 


(notice_ author, board _ name, notice _ music, notice _ title, notice _ content, notice _ date) values 


?,?,?,?,?,now())”); 
pstmt. setString(1, 
pstmt. setString(2, 
pstmt. setString(3, 
pstmt. setString(4, 


.getNotice Author()) ; 
.getBoard Name()); 
.getNotice Music()) ; 
.getNotice Title()) ; 
pstmt. setString(5, a.getNotice Content()) ; 


Popp 


pstmt. executeUpdate() ; 
pstmt. close() ; 
conn. close(); 
) 
catch( SQLException e) í 
System. out. println(e) ; 
) 
) 
) 


DelnoticeBean 业务 模型 的 主要 功能 是 : 删除 公告 。 
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代码 模板 DelnoticeBean. java 如 下 : 


package net.icefish. bean; 
import java. sql. Connection; 
import java. sql. PreparedStatement; 
import java. sql. SQLException; 
import net. icefish. bean. Conn; 
public class DelnoticeBean ( 

private Connection conn; 


public DelnoticeBean() | 

this. conn— Conn. connection() ; 
) 
public void delBe( NoticeBean a) ( 

PreparedStatement pstmt= null; 

try{ 

String xg=a. getNotice_ID(); 

String sql= "delete from icefish notice where notice id in("+xg+")"; 
pstmt. executeUpdate( sql) ; 
pstmt. close() ; 
conn. close) ; 

) 

catch( SQLException e) | 

System. out. println(e) ; 

) 

) 
) 


HelpDataBean 业务 模型 的 主要 功能 是 : 增加 帮助 修改 帮助 和 删除 帮助 。 
代码 模板 HelpDataBean. java 如 下 : 


public class HelpDataBean { 
private Connection conn; 


public HelpDataBean() | 
this. conn— Conn. connection() ; 
) 
// 增 加 新 帮助 
public boolean addHelp( HelpBean helpBean) ( 
boolean flag= false; 
PreparedStatement pstmt1 — null; 
try{ 
pstmtl 一 conn. prepareStatement ( " insert into icefish _ help ( help _ name, help _ info) 
values(?,?)"); 
pstmtl.setString(1, helpBean. getHelp_Name()); 
pstmtl.setString(2, helpBean.getHelp Info()); 
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int result] = pstmt] .executeUpdate( ); 
if (resultl > 0) í 
flag = true; 
) 
else( 
flag — false; 
) 
pstmtl.close() ; 
conn. close) ; 
) 
catch( SQLException e) | 
flag false; 
System. out. println(e. toString()) ; 


) 

return flag; 
) 
// 修 改 帮 助 内 容 


public boolean editHelp( HelpBean helpBean) ( 
boolean flag= false; 
PreparedStatement pstmtl = null; 
try( 
pstmtl = conn. prepareStatement ( " update icefish help set help_name=?, help info—? 
where help id—"--helpBean. getHelp ID(); 
pstmtl.setString(1, helpBean. getHelp Name()); 
pstmtl.setString(2, helpBean. getHelp Info()) ; 
int result] = pstmtl . executeUpdate() ; 
if (resultl > 0)( 
flag — true; 
) 
else( 
flag — false; 
) 
pstmtl.close() : 
conn. close() ; 
) 
catch( SQLException e) | 
flag false; 
System. out. println(e. toString()) ; 
) 
return flag; 
) 
// 删 除 帮 助 内 容 
public boolean delHelp( HelpBean helpBean) ( 
boolean flag- false; 
PreparedStatement pstmt1 — null; 
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try{ 
pstmtl — conn. prepareStatement( "delete from icefish help where help id—?"); 
pstmtl.setString(1, helpBean. getHelp IDO); 
int result] = pstmtl . executeUpdate( ) ; 
if (resultl > 0) í 

flag = true; 
) 
elset 
flag — false; 

) 
pstmtl.close() ; 
conn. close) ; 

) 

catch( SQLException e) | 
flag false; 
System. out. println(e. toString()) ; 

) 

return flag; 

) 
) 


InsertmyfriendBean 业务 模型 的 主要 功能 是 : 在 论坛 中 添加 好 友 。 
代码 模板 InsertmyfriendBean. java 如 下 ; 


package net. icefish. bean; 

import java. sql. * ; 

import net. icefish. bean. Conn; 

public class InsertmyfriendBean ( 
private Connection conn; 


public InsertmyfriendBean() | 
this. conn— Conn. connection) ; 
) 
public void tjBe(UserBean a) | 
PreparedStatement pstmt— null; 
PreparedStatement pstmt2— null; 
try{ 
pstmt 一 conn. prepareStatement("insert into icefish_friend 
(user_id, friend_id, friend_name, friend_addtime, friend_Email) value(?,?,?,now(),?)"); 
pstmt. setString(1, a. getUser_IDO); 
pstmt. setString(2, a. getFriend_ID(C)) ; 
pstmt. setString(3, a. getFriend_Name()); 
pstmt. setString(4, a.getFriend Email()); 
pstmt. executeUpdate() ; 
pstmt. close() ; 
conn. close) ; 
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} 
catch(SQLException e) í 
System. out. println(e) ; 
} 
) 
) 


MymodifyBean 业务 模型 的 主要 功能 是 : 用 户 个 人 信息 修改 。 
代码 模板 MymodifyBean. java 如 下 : 


package net.icefish. bean; 
import java. sql. * ; 

import net. icefish. bean. Conn; 
public class MymodifyBean í 
private Connection conn; 


public MymodifyBean() | 
this. conn— Conn. connection() ; 


public void xgBe(UserBean a) | 

PreparedStatement pstmt— null; 

try( 

pstmt= conn. prepareStatement( "update icefish user set user_secondname 一 ?,user_sex 一 ?， 


user birthday— ?, user QQ=?, user group—?, user face— ?, user sign— ?, user truename— ?, 
user Email— ?, user tel —?, user blood —?, user shengxiao 一?, user nation—?, user province ?, 
user city—? where user id—?"); 

pstmt. setString(1, a. getUser Secondname()) ; 

pstmt. setString(2, a. getUser Sex()); 

pstmt. setString(3, a. getUser Birthday()): 

pstmt. setString(4, a.getUser QQO); 

pstmt. setString(5, a. getUser Group()) : 

pstmt. setString(6, a. getUser Face()); 

pstmt. setString(7, a.getUser Sign()); 

pstmt. setString(8, a. getUser Truename()) ; 

pstmt. setString(9, a. getUser Email()) ; 

pstmt. setString( 10, a.getUser Tel()); 

pstmt. setString(11, a. getUser Blood() ; 

pstmt. setString(12, a.getUser Shengxiao()); 

pstmt. setString(13, a.getUser Nation()); 

pstmt. setString(14, a.getUser Province()); 

pstmt. setString( 15, a. getUser City()); 

pstmt. setString( 16, a.getUser IDO); 

pstmt. executeUpdate() ; 

pstmt. close() ; 

conn. close) ; 


} 
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catch( SQLException e){ 
System. out. println(e) ; 
) 
) 
) 


RepasswordBean 业务 模型 的 主要 功能 是 : 修改 密码 。 
代码 模板 RepasswordBean. java 如 下 : 


import java.sql. * ; 

import net. icefish. bean. Conn; 

public class RepasswordBean | 
private Connection conn; 


public RepasswordBean() ( 
this. conn— Conn. connection() ; 

) 

public void mmBe(UserBean a) ( 
PreparedStatement pstmt— null; 
try{ 
pstmt= conn. prepareStatement( "update icefish user set user_password=? where user id—?"); 
pstmt. setString(1, a. getUser Password()); 
pstmt. setString(2, a. getUser IDO); 
pstmt. executeUpdate() ; 
pstmt. close() ; 
conn. closeO ; 
) 
catch( SQLException e) | 
System. out. println(e) ; 
) 

) 

) 


LinkDataBean 业务 模型 的 主要 功能 是 : 添加 友情 链接 、 修 改 友 情 链 接 和 删除 友情 
链接 。 
代码 模板 LinkDataBean. java 如 下 : 


public class LinkDataBean ( 

private Connection conn; 

public LinkDataBean() ( 
this. conn Conn. connection( ) ; 

) 

// 增 加 新 链接 

public boolean addLink(LinkBean linkBean) ( 
boolean flag= false; 
String link pass— new String() ; 
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if(linkBean. getLink_Pass()){ 
link_pass= "true"; 
}else{ 
link_pass= "false"; 
) 
String link islogo— new String() ; 
ifClinkBean. getLink Islogo()) í 
link islogo— "true"; 
Jelse( 
link islogo- "false" ; 
) 
PreparedStatement pstmtl = null; 
try{ 
pstmtl — conn. prepareStatement( "insert into icefish link 


(link name,link info,link url,link logo,link islogo,link pass) values(?, ?,?,?,?, ?)"); 


) 


pstmtl.setString(1, linkBean. getLink Name()); 
pstmtl.setString(2, linkBean. getLink Info()); 
pstmtl. setString(3, linkBean.getLink Url()) ; 
pstmtl.setString(4, linkBean. getLink Logo()); 
pstmtl.setString(5, link islogo) ; 
pstmtl.setString(6, link pass); 
int result] = pstmtl . executeUpdate() ; 
if (resultl > 0)( 
flag — true; 
) 
else{ 
flag = false; 
} 
pstmtl. close() ; 
conn. close() ; 
) 
catch( SQLException e) | 
flag— false; 
System. out. printIn(e. toString()) ; 
) 


return flag; 


// 修 改 链接 内 容 
public boolean editLink(LinkBean linkBean) ( 


boolean flag- false; 
String link pass— new String() ; 
if(linkBean. getLink Pass()( 
link pass— "true"; 
Jelse( 
link pass— "false"; 
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) 
String link islogo— new String() ; 
if(linkBean. getLink Islogo()) ( 
link islogo— "true"; 
Jelse( 
link islogo- "false" ; 
) 
PreparedStatement pstmtl = null; 
try( 
pstmtl — conn. prepareStatement( "update icefish_link set link name— ?, link info—?, 
link url—?,link logo -—?,link islogo —?, link_ pass =? where link id =" + linkBean 
.getLink IDO); 
pstmtl.setString(1, linkBean. getLink NameO); 
pstmtl.setString(2, linkBean. getLink Info()) ; 
pstmtl.setString(3, linkBean. getLink Url()) ; 
pstmtl.setString(4, linkBean. getLink Logo(); 
pstmtl.setString(5, link islogo) ; 
pstmtl.setString(6, link pass); 
int result] = pstmtl.executeUpdate() ; 
if (resultl > 0) ( 
flag = true; 
) 
else( 
flag — false; 
) 
pstmtl. close) ; 
conn. close(); 
) 
catch( SQLException e) | 
flag= false; 
System. out. println(e. toString()) ; 
) 
return flag; 
) 
// 删 除 链接 内 容 
public boolean delLink(LinkBean linkBean) ( 
boolean flag- false; 
PreparedStatement pstmt1 = null; 
try{ 
pstmt1 一 conn. prepareStatement( "delete from icefish link where link id—?"); 
pstmtl.setString(1, linkBean. getLink ID()); 
int result] = pstmtl. executeUpdate() ; 
if (resultl > 0) í 
flag = true; 
) 
elset 
flag — false; 
) 
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pstmtl. close) ; 
conn. close) ; 

) 

catch( SQLException e) í 
flag false; 


System. out. println(e. toString()) ; 
) 


return flag; 


12.5 系统 实现 
12.5.1 用 户 注册 


当 用 户 注 册 时 ,该 模块 要 求 用 户 接受 注册 声明 ,然后 输入 用 户 名 性别、 密码 等 信息 ,和 否 
则 不 允许 注册 。 注 册 成 功 的 用 户 信息 被 存 人 数据 库 的 icefish_user 表 中 。 

1. 视图 (JSP 页 面 ) 

该 模块 中 有 三 个 JSP 页 面 : reg. jsp.adduser. jsp 和 regsuccess. jsp。reg. jsp 显示 注册 
说 明 , 需 要 用 户 接 受 方 可 进行 后 续 的 注册 ; adduser. jsp 提供 注册 信息 输入 界面 ; regsucess. jsp 
提示 用 户 注册 是 否 成 功 。 


代码 模板 reg. jsp 代码 如 下 ,效果 如 图 12-4 所 示 。 


Cd 
— ! JUST ES CRIT RO 
2> ——— 
MAE. MESAI EN TE 
> Tubi ini 


O 冰 鱼 论坛 ~ 论坛 注册 > 注册 声明 


服务 条 未 和 声明 


欢迎 您 加 入 【 冰 鱼 论坛 】 参加 交流 和 讨论 ,， 【 冰 鱼 论坛 ] 为 公共 论坛 ， 为 维护 中 上 公共 秩序 和 社会 稳定 ， 请 您 自觉 道 守 以 下 条 款 : 
[- 、 不 得 利用 本 站 危害 国家 安全 、 池 需 国 家 秘密 ， 不 得 侵犯 国家 社会 集体 的 和 公民 的 合法 权益 ， 不 得 利用 本 站 制作 、 复 制 和 传播 下 列 信息 : 


(二 ) 炉 动 抗拒 USA 行政 法 规 实施 的 # 
zh 二 cow 


crorpi mill 1 
SEE 
E68 
PE 
bier 
La 
ise 
n. 


nies DE cam PEU 


律 行 政 
aaa 


38 


Ln 
| 二 、 互相 尊重 ， 对 自己 的 言论 和 行为 负责 。 


我 接受 


12-4 注册 声明 页 面 
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<% (8 page pageEncoding= "gb2312" ⁄— 
<% (8 page contentType— "text/html;charset— gb2312" 947» 
<% (9 include file— "top.jsp" $47» 
<html> 
<head> 
<title> VK fa i£ Tz-— 63s t WE AR 23 pd x — / itle 
<script Language- "Javascript" src= "include/form.js" >< / Script 
</head> 
<body style= "text-align: center"> 
<div align= left>>> 欢迎 光临 <b> fa 161 — / b -—/div 
< table width =" 100%" height =" 30" border =" 0" bordercolorlight =" it 7777ff" 
bordercolordark= " # 7777ff" style= "border-collapse: collapse" > 
<tr bgcolor=" # dff2ed"> <td><img src= "images/skin/1/forum_nav. gif"> 
<a class=zh href=""> Kk EI </a> 一 
<a class 一 zh href 一 "reg.jsp" 之 论坛 注册 一 /a> 一 注册 声明 一 /td>> 二 /tr> 
</table> 
<form method="post" name= "myform" action= "adduser. jsp?reg— apply"> 
< table width =" 100%" border ="1" bordercolorlight =" # 7777ff" bordercolordark = " # 
7777ff" style= "border-collapse: collapse" > 
<tr> 
<td height= "25" align= "center" background = "images/skin/1/bg_td. gif" > <b> 
<font color=" # ffffff"— JR # RAAE B] — / font </b> </td> 
<s> 
<tr> 
二 td 二 <b 二 继续 注册 前 请 先 阅 读 [ 冰 鱼 论坛 论坛 协议 二 /b 二 <p 二 
欢迎 您 加 入 [ 冰 鱼 论坛 参加 交流 和 讨论 ,【 冰 鱼 论坛 ] 为 公共 论坛 ， 
为 维护 网 上 公共 秩序 和 社会 稳定 ,请 您 自觉 遵守 以 下 条 款 : <p> 
一 、 不 得 利用 本 站 危害 国家 安全 、 泄 露 国家 秘密 ,不 得 侵犯 国家 社会 集体 的 和 
公民 的 合法 权益 ,不 得 利用 本 站 制作 、 复 制 和 传播 下 列 信息 : p 
(一 ) 煽动 抗拒 \ 破 坏 宪法 和 法 律 , 行 政法 规 实施 的 ;二 br 二 
(=) fi ah i 家 政权 ,推翻 社会 主义 制度 的 ;二 br 
(三 ) 煽动 分 家 、 破 坏 国家 统一 的 ;一 br> 
(四 ) 煽动 民族 仇恨 、 民 族 歧视 ,破坏 民族 团结 的 ;一 br> 
(五 ) 捏造 或 者 焉 曲 事实 ,散布 谣言 ,扰乱 社会 秩序 的 ;一 br> 
(六 ) 宣扬 封建 迷信 和 淫秽、 色情、 赌博 、 暴 力 、 凶 杀 、 恺 怖 、 教 唆 犯 罪 的 ;二 br 二 
(E) 公然 侮辱 他 人 或 者 捏造 事实 诽谤 他 人 的 ,或 者 进行 其 他 恶意 攻击 的 ;二 br> 
CA) 损害 国家 机 关 信 誉 的 ;br> 
Qv 其 他 违反 宪法 和 法 律 行政 法 规 的 ;一 br> 
(十 ) 进行 商业 广告 行为 的 . p 
二 、 互相 尊重 ,对 自己 的 言论 和 行为 负责 . </td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td align= "center" ><input type="submit" value= "REZ" ></td> 
</tr> 
</table> 
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</form> 
</body> 
</html> 


代码 模板 adduser. jsp 如 下 ,效果 如 图 12-5 所 示 。 


4». 此 处 可 放置 广告 《 顶部 广告 ) 


> ets xtti 
o kenit — 论坛 主 册 一 注册 新 用 户 
新 用 户 注 册 
用 户 各: 
主 册 用 户 名 长度 限制 为 ~ sot tang [ILES 
[r3 
请 法 择 您 的 性 别 ef oft«« 
EB: GERARD 
WA: EDXNS- WASAN hi ”或 ma FF m 
WEB: GERARD 
JA ANA ALA E 
EI 
请 办 入 忘记 密码 的 提示 问 是 tane 
MRSE: i = 
XSCEPHMPTISERUR- 用 于 取 加 论坛 密友 pane 
Emai! a 
LI qm uj x 
[es] 


12-5 注册 信息 填写 页 面 


<% G page pageEncoding= "gb2312" % > 
<% (9 page contentType— "text/html;charset— gb2312" % > 
<% (9 include file— "top.jsp" 947» 
<html> 
<head> 
过 title 二 冰 鱼 论坛 -一 -论坛 注册 新 用 户 过 /title 二 
<script Language= "Javascript" src= "include/form. js"></Script> 
</head> 
<body style=" text-align: center" > 
<div align=left> >> 欢迎 光临 <b> fa ie — /b></div> 
< table width =" 100%" height =" 30" border =" 0" bordercolorlight =" # 7777ff" 
bordercolordark= " # 7777ff" style= "border-collapse: collapse" > 
<tr bgcolor=" # dff2ed"> 
<td> 
<img src= "images/skin/1/forum_nav. gif" > 
<a class 一 zh href=""> áa </a> 一 
<a class 一 zh href 一 "reg.jsp" 之 论坛 注册 一/a> > 注册 新 用 户 
</td> 
</tr> 
</table> 
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<% 
String reg= request. getParameter("reg") ; 
if(reg! =null) ( 
if(reg.equals("apply")) í 
> 
<form method="post" name= "adduser" action=" AddUserServlet" > 
< table width =" 100%" vlign =" center" border =" 1" bordercolorlight =" # 7777íf" 
bordercolordark= " # 7777ff" style= "border-collapse: collapse" > 
<tr> 
<td colspan= 
<b> 
<font color=" # ffffff"— š P! kW < / font 
</b> 
</td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td width="40%" height="35"> <b> JH P4: </b><br> 
注册 用 户 名 长 度 限制 为 0 一 50 字 节 


height— "25" align="center" background= "images/ skin/ 1/ bg. td. gif" > 


</td> 
<td> &-nbsp; <input type="text" name= "user. name" size= "24" maxlength= "50"> &-nbsp; 
<input name=" Submit3" type="button" value 二 "检测 用 户 名 " onClick— CheckUser()77 


</td> 
</tr> 
<tr> 
<td width="40%" height "35" <b> tE 别 : </b> <br> Hii f£ fas i fk 9) </td> 
<td> 
<input type— "radio" name= "user. sex" value=" " checked> 
<img src= "images/male. gif" > Il RF 
<input type= "radio" name= "user sex" value= "# "><img src= "images/ female. gif" >— 3: dr 
</td> 
</tr> 


<tr bgcolor=" # dff2ed"> 
<td width="40%" height="35"> <b>% 码 : (E/PA BL) </b> 
过 br 二 请 输入 密码 ,区 分 大 小 写 .请 不 要 使 用 任何 类 似 '* '、' ' 或 HTML 字符 


</td> 

<td> 
<input type— "password" name= "user password" size= "26" maxlength= "50" ></td> 
</tr> 

<tr> 
<td width="40%" height="35"> <b> MURD: (EDAK) </b><br> 

请 再 次 输入 密码 以 便 确 认 

</td> 

<td> 


<input type— "password" name= "user password2" size= "26" onchange— CheckPass() ></td> 
xu 
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«tr bgcolor=" # dff2ed"> 
<td width="40%" height="35">><b> $W: < /b>—<br> 
请 输入 忘记 密码 的 提示 问题 
</td> 
<td> 
<input type="text" name= "user_password_q" size="24"> 
</td> 
</tr> 
<tr> 
<td width="40%" height="35"> <b> MEER: </b><br> 
忘记 密码 的 提示 问题 答案 ,用 于 取 回 论坛 密码 
</td> 
<td> 
<input type="text" name= "user_password_a" size="24"> </td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td width="40%" height= "35">>—b> Email 地 址 : </b><br> 
请 输入 有 效 的 邮件 地 址 ,这 将 使 您 能 用 到 论坛 中 的 所 有 功能 
</td> 
<td> 
<input type="text" name= "user_Email" size="24"></td> 
</tr> 
<tr> 
<td colspan= "2" height="25" align="center" background= "images/skin/1/bg_td. gif"> 
<input type="submit" value 二 "注册 " onclick— CheckReg() > 
<input type= "reset" value=" HH" ></td> 
</tr> 
</table> 
</form> 
<% 


} }else{ 
%> 
<form method="post" action— "reg.jsp"> 
< table width =" 100%" border =" 1" bordercolorlight =" # 7777ff" bordercolordark = 
"#7777ff" style= "border-collapse: collapse" > 
<tr> 
<td height= "25" align="center" background= "images/skin/1/bg td. gif" ><b> 
<font color=" # ffffff"— JR # RAAE PA — / font — / b 
</td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td height="80" align="center"> KRA 8224618 WEBS Ct 46 RKA B), 
请 返回 论坛 声明 页 面 接受 方 能 进行 注册 .一 p> 点 击 下面 按 钮 可 返回 注册 声明 页 面 
</td> 
</tr> 
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<tr> 
<td height= "25" align="center" background= "images/skin/1/bg_td. gif" > 
<input type="submit" value="iE 回 "之 
</td> 

</tr> 
</table> 
</form> 
<%}%> 

</body> 
</html> 


代码 模板 regsucess. jsp 如 下 ,效果 如 图 12-6 所 示 。 


4» 此 站 有 加 广 告 (顶部 广告) 
< 


对 不 起 ,你 还 没有 【〖 注册 了 或 〖 登录 〗， 因 此 无 法 看 到 用 户 资料 管理 菜单 


>> mid ibit 
9 水 和 论坛 > 论坛 主 肌 一 注册 成 功 


注册 成 由 : 冰 鱼 论坛 欢迎 您 的 到 来 ! 


OSBüs 注册 成 功 ,请 点 击 这 里 返回 论坛 首页 进行 登录 。 


12-6 注册 提示 页 面 


<% @page pageEncoding= "gb2312" % > 
<% @ page contentType— "text/html;charset— gb2312" %> 
<% @include file— "top. jsp" % > 
<html> 
<head> 
<title> VK f& ie $z- — EE — / title 
<script Language "Javascript" src— "include/form.js" >< /Script2> 
</head> 
<body style= "text-align: center" > 
<div align= eft^» 欢迎 光临 <b> fa ie 1E— /b></div> 
< table width =" 100%" height =" 30" border =" 0" bordercolorlight =" # 7777ff" 
bordercolordark= " # 7777ff" style= "border-collapse : collapse" > 
<tr bgcolor=" # dff2ed"> 
<td><img src= "images/skin/1/forum nav. gif" > 
<a class=zh href=""> f&ái£Tz </a> 一 
<a class=zh href="reg.jsp"> i£izikHb- /a— > 注册 成 功 
</td> 
</tr> 
</table> 
<% 
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String reg— request. getParameter("reg") ; 
if(reg! =null) ( 
if(reg. equals("ok")) ( 
X> 
<form method="post" name= "adduser" action= "index.jsp"— 
< table width =" 100%" border =" 1" bordercolorlight — " # 7777ff" bordercolordark = 
"#7777ff" style= "border-collapse: collapse" > 
<tr> 
<td height= "25" align="center" background= "images/skin/1/bg_td. gif" ><b> 
<font color=" # fff" > HRD: vk f& ie dz XUL f: üJ AAK! < / font < / b 
</td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td height= "80" align= "center" ><a class=zh href= "index. jsp"> 
Ii 28 E NEID f 8 hx HE [lie Tn K ETER. / li </a> 
</td> 
</tr> 
<tr> 
<td colspan="2" height="25" align="center" background= "images/ skin/ 1/ bg. td. gif" 
</td> 
</tr> 
</table> 
<form> 
<% 
}}else{ 
> 
<form method="post" action= "reg.jsp"2> 
< table width =" 100%" border =" 1" bordercolorlight =" # 7777ff" bordercolordark = 
"#7777ff" style= "border-collapse: collapse" > 
«tu 
<td height— "25" align="center" background— "images/skin/1/bg td. gif" ><b> 
<font color=" # ffffff" > 1E EA W !< / font — / b 
</td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td height= "80" align="center"> 
你 没有 成 功 注 册 , 原 因 可 能 是 你 注册 的 用 户 名 已 存在 ， 
也 可 能 是 你 没有 正确 地 提交 注册 信息 . 
< 去 p> 点 击 下 面 按钮 返回 注册 声明 页 面 
</td> 
</tr> 
> 
<td height= "25" align= "center" background= "images/skin/1/bg_td. gif" > 
< input type="submit" value= "i& [s| "> 
</td> 


274 网 络 编程 与 动态 网 站 制作 


</tr> 
</table> 
</form> 
WW 
</body> 

</html> 

2. 控制 器 (Servlet) 

用 户 注 册 模 块 所 涉及 到 的 控制 器 Servlet 是 AddUserServlet 和 CheckUser( 见 12. 3. 4 
节 的 web. xml 配置 文件 )。AddUserServlet 控制 器 获取 视图 的 请 求 后 ,将 视图 中 的 信息 封 
装 在 实体 模型 UserBean( 见 12. 4. 2 节 ) 中 。 如 果 获 取 的 请 求 是 检查 用 户 名 是 否 可 用 , 则 调 
用 CheckUser 中 的 方法 来 进行 判断 ,并 弹出 提示 对 话 框 。 不 管用 户 名 是 否 存在 ,都 会 弹出 
对 应 的 提示 窗口 ,如 图 12-7 所 示 。 


http://localhost8080/icefish/CheckL Ë http://localhost:8080/icefish/Checkt. Ë 
| 
y 对 不 起 ! 该 用 户 名 已 经 存在 ， 不 
KE! 该 用 户 名 可 以 注册 ! 能 进行 注册 1 


12-7 检测 操作 反馈 信息 
代码 模板 AddUserServlet. java 如 下 : 


public class AddUserServlet extends HttpServlet { 
public void init(ServletConfig config) throws ServletException ( 
super. init(config) ; 
) 
public void destroy() { 
super. destroy () ; 
) 
protected void doGet( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
doPost(request, response) ; 
) 
protected void doPost( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException { 
response. setContentT ype( "text/html ; charset — GB2312") ; 
request. setCharacterEncoding( " GB2312") ; 
PrintWriter out= response. getWriter() ; 
HttpSession session— request. getSession(true) ; 


String user name— request. getParameter("user name"); 
String user password request. getParameter("user password"); 
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String user sex— request. getParameter("user sex"); 

String user password q- request. getParameter("user password q"); 
String user password a— request. getParameter("user password a"); 
String user. Email request. getParameter("user Email"); 


UserDataBean udb— new UserDataBean() ; 
UserBean userBean— new UserBean() ; 
userBean. setUser Name(user name); 
userBean. setUser. Password(user password) ; 
userBean.setUser Password a(user password a); 
userBean.setUser Password q(user password q); 
userBean. setUser Email(user Email) ; 
userBean. setUser Sex(user sex); 
boolean checkUser— udb. checkUser(userBean) ; 
if(checkUser) ( 
boolean result = udb.addUser(userBean) ; 
RequestDispatcher rd = request. getRequestDispatcher( " regsuccess. jsp?reg— ok") ; 
rd. forward(request, response) ; 
) 
else( 
RequestDispatcher rd = request. getRequestDispatcher( " regsuccess. jsp") ; 
rd. forward(request, response); 
) 
Jelse( 
RequestDispatcher rd = request. getRequestDispatcher( " regsuccess. jsp") ; 
rd. forward( request, response); 


) 
) 


代码 模板 CheckUser. java 如 下 : 


public class CheckUser extends HttpServlet { 
public void init(ServletConfig config) throws ServletException ( 
super. init(config) ; 
) 
public void destroy() ( 
super. destroy () ; 
) 
protected void doGet( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
doPost(request, response) ; 
) 
protected void doPost( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
response. setContentType( "text/html;charset— GB2312") ; 
request. setCharacterEncoding(" GB2312") ; 
PrintWriter out— response. getWriter() ; 
HttpSession session request. getSession( true) ; 
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String user_name 一 request. getParameter("user name"); 

UserDataBean udb— new UserDataBean() ; 

UserBean userBean— new UserBean() ; 

userBean.setUser Name(user name); 

boolean checkUser— udb. checkUser(userBean) ; 

ifCcheckUser) í 

out. println("< table width=100% height— 10024 align=center vlign— center border—1 

bordercolorlight— # 7777ff bordercolordark- # 7777ff style= border-collapse: collapse " 
+"<tr bgcolor= # dff2ed align= center <td><font color blue 
恭喜 ! 该 用 户 名 可 以 注册 !<</font>>< 王 /td> 二 /td> " 4- "< /table"); 


)else( 
out. println("<table width=100% height=100% align=center vlign= center border=1 
bordercolorlight= # 7777ff bordercolordark= # 7777ff style= border-collapse: collapse>>" 
+"<tr bgcolor= # dff2ed align= center> <td><font color=blue> 
对 不 起 ! 该 用 户 名 已 经 存在 ,不 能 进行 注册 !</font>></td></td>"+"</table>"); 


) 
) 


如 果 获 取 的 请 求 是 检查 密码 等 信息 是 否 为 空 , 则 调用 form. js( JavaScript 页 面 ) 中 的 方 
法 进行 检测 form. js 针对 该 功能 的 代码 如 下 (由 于 form. js 还 有 其 他 功能 ,所 以 在 此 并 未 
列 出 其 全 部 代码 ) : 


// 注 册页 面 检测 两 次 密码 是 否 输入 相同 
function CheckPass() 
{ 
if(document. adduser. user. password. value= = '' | | document. adduser. user_password2. value= = 


X 


alert(" 你 还 没有 输入 密码 ,请 输入 密码 !"); 
window. location. reload() ; 
)elseif( document. adduser. user. password. value! = document. adduser. user. password2. value) | 
alert(" 两 次 输入 的 密码 不 一 样 , 请 重新 输入 !") ; 
window. location. reload() ; 
) 
) 


// 检 测 用 户 名 是 否 已 存在 
function CheckUser() 
{ 
if( document. adduser. user name. value= = '') | 
alert(" 你 还 没有 输入 用 户 名 ,请 输入 用 户 名 !"); 
}else{ 


window. open( "CheckUser? user_name =" + document. adduser. user. name. value, "", " width = 
250, height= 150"); 
) 
) 
// 提 交 注 册 信 息 时 的 验证 
function CheckReg() 
4 
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if(document. adduser. user. name. value = = '' | | document. adduser. user. password. value = = ''| | 
document. adduser. user_password2. value= = '') | 
alert(" 你 还 没有 输入 用 户 名 或 者 密码 ,请 输入 用 户 名 或 者 密码 !"); 
window. location. reload() ; 
}else if(document. adduser. user password. value! = document. adduser. user password2. value) ( 
alert(" 两 次 输入 的 密码 不 一 样 ,请 重新 输入 ! ") ; 
window. location. reload() ; 
}else if (document. adduser. user. password a. value = = '' | | document. adduser. user. password. q 
. value= = '*) ( 
alert(" 请 填 人 密码 问题 和 密码 答案 ,方便 以 后 取 回 论坛 密码 1"); 
window. location. reload() ; 
} 
} 
如 果 获 取 的 请 求 是 注册 ,AddUserServlet 则 调用 UserDataBean( 见 12. 4. 3 节 ) 业 务 模 
型 中 的 adduser() 等 方法 进行 注册 。 注 册 完 成 后 进入 regsuccess. jsp 页 面 ,提示 用 户 注 册 成 
功 或 注册 失败 ,显示 效果 如 前 面 的 图 12-5 所 示 。 如 果 成 功 注册 , 则 转 到 index. jsp 页 面 ,可 
以 登录 系统 ; 如 果 失 败 , 则 转 到 reg. jsp 页 面 ,继续 重新 注册 。 


12.5.2 用 户 登 录 


用 户 输入 自己 的 用 户 名 和 密码 以 及 验证 码 之 后 ,论坛 系统 将 对 用 户 名 和 密码 进行 验证 。 
如 果 用 户 名 和 密码 以 及 验证 码 同 时 正确 , 则 登录 成 功 ,进入 系统 主 界面 ; 如 果 用 户 名 、 密 码 
或 者 验证 码 有 误 , 则 回 到 登录 页 面 继续 重新 登录 。 

1. 视图 (JSP) 

页 面 login. jsp 提供 登录 信息 输入 界面 ,效果 如 图 12-8 所 示 。 


— a KATEA (SrA 
ATE PERAI S IRI SZ EO 
> TRID kt 


O 冰 鱼 论坛 > 论坛 登录 + 填写 登录 信息 
请 输入 您 的 用 户 名 、 论 坛 密码 登录 


| 清 输 入 您 的 用 户 名 没有 注册 ? 
| 清 输 入 您 的 论坛 密码 忘记 密码 ? 
| 请 输入 验证 码 rs 

图 不 保存 ， 关 闭 浏览 器 就 失效 
[Cookie 选项 口 保存 -天 
[ERE RU Cookie 保存 时 间 ， 下 次 访问 可 以 方便 给 入 。 ORA 

Ofr 

až EE] 


图 12-8 登录 界面 
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代码 模板 login. jsp 如 下 : 


<% @ page pageEncodini gb2312"%> 
<% @ page contentType= "text/html; charset= gb2312" %> 
<% @include file 
<% @page import= "java. util. * " % > 
<html> 
<head> 
<<title> 冰 鱼 论坛 -一 -论坛 注册 新 用 户 二 /title> 
<script Language= "Javascript" src= "include/form.js"></Script> 
</head> 
<% 
// 取 随 机 产生 的 认证 码 (4 位 数字 ) 
String sRand= 
// 生 成 随机 类 
Random random = new Random() ; 
for (int i=0;i<4;i++){ 
String rand— String. valueOf( random. nextInt(10)) ; 
sRand-- — rand; 
) 
// 将 认证 码 存 人 SESSION 
session. setAttribute( "rand", sRand) ; 
n 
<body style= "text-align: center" 
<div align— left > 欢迎 光临 <b> v f&ie Ez — / b —/ div 
< table width =" 100%" height =" 30" border =" 0" bordercolorlight =" # 7777íf" 
bordercolordark— " # 7777ff" 
style— "border-collapse: collapse" > 
<tr bgcolor=" # dff2ed" 
<td><img src= "images/skin/1/forum nav. gif" 
<a class—zh href=""> fai£iz-—/a— 一 
<a class= zh href 二 "login.jsp" 祖 论坛 登录 二 /a 二 > 填写 登录 信息 
</td> 
</tr> 
</table> 
<form method="post" name= "userlogin" action= "check. jsp" 
<table width="100%" vlign— "center" border— "1" bordercolorlight— " # 7777ff" 
bordercolordark— " # 7777ff" style= "border-collapse: collapse" > 
<tr> 
<td colspan="2" height="25" align="center" background= "images/ skin/ 1/bg td. gif"> 
«b «font color=" # fff" >A A fs ff] Fl P 44 ME ERES T / font —/ b 
</td> 
< r> 
<tr bgcolor=" # dff2ed"> 
<td width="30%" height="35" >W f A fs f FH P1 4 </td> 
<td><input type="text" name= "user name" size— "24" 
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<a class= zh href— "reg. jsp" ^ CB EE? </a> 
</td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td width="30%" height= "35" >i fh A K HEI 8r 83 < / td> 
<td><input type="password" name= "user_password" size="26"> 
<a class 一 zh href=""> BRE? </a> 
</td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td width="30%" height= "35" ifi $8 A RIEM: </td> 
«td input name= "checknumber" type="text" maxlength= "4" size="6"> 
< out. print(sRand) ; X> 
</td> 
</tr> 
<tr bgcolor=" # dff2ed"> 
<td width= "30% "><b>Cookie 选项 一 /b> 
<br> HAERA Cookie 保存 时 间 , 下 次 访问 可 以 方便 输入 . 
</td> 
<td> 
<input type="radio" name= "save login" value= "no_time" checked 
不 保存 ,关闭 浏览 器 就 失效 
<brœ< input type="radio" name= "save login" value— "one day" ff f£— X 
—br- input type="radio" name= "save login" value— "one month" fi f£ — A 
«br input type= "radio" name= "save login" value— "one year" > f f£ — F 
</td> 
</tr> 
<tr> 
<td colspan="2" height="25" align="center" background= "images/ skin/ 1/bg td. gif"> 
<input type="submit" onclick— "CheckLogin()" value=" % "> 
<input type= "reset" value— "# "> 
</td> 
</tr> 
</table> 
</form> 
</body> 
</html> 


2. 控制 器 (Servlet) 

用 户 登 录 模 块 所 涉及 的 控制 器 Servlet 对 象 的 名 称 是 LoginServlet( 见 12. 3. 4 节 中 的 
web. xml 配置 文件 ) 。 控 制 器 获取 视图 的 请 求 后 ,将 视图 中 的 信息 封装 在 实体 模型 User( 见 
12.4. 2 节 ) 中 ,然后 调用 业务 模型 UserDataBean( 见 12. 4. 3 节 ) 中 的 登录 方法 执行 相应 的 业 
务 处 理 。 登 录 成 功 进 入 系统 主 界面 (如 图 12-9 所 示 ) ,登录 失败 则 返回 login. jsp 页 面 。 
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oL s, 
[< HARAN E CERT RO 
B 


ee 
t 重 登录 | MARIER | AHER | mE) | ERS | 我 的 好 友 | 查看 论坛 用 户 | 论坛 后 自 管 理 | 退出 


此 处 可 放置 广告 《首页 大 幅 广 告 ) 


ws 


[MARO Gr 
D TRMA JUM CHEER 


版 主 : 龙 龙 


专区 
?情感 护 过 时 空 的 界线 ， 你 我 相 蛇 在 这 里 。 曾 经 伤心 的 你 ， 今 天 还 好 3? ! 
版 主 : 龙 龙 


12-9 系统 主 界面 
代码 模板 LoginServlet. java 如 下 : 


package net. icefish. servlet; 
import net. icefish. bean. * ; 
import java.net. * ; 
import java. io. IOException; 
import java. io. PrintWriter; 
import javax.servlet. * ; 
import javax. servlet. http. * ; 
public class LoginServlet extends HttpServlet ( 
public void init(ServletConfig config) throws ServletException ( 
super. init(config) ; 
) 
public void destroy() { 
super.destroy(); 
) 
protected void doGet( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
doPost(request, response) ; 
) 
protected void doPost( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
response. setContentType(" text/html ;charset— GB2312") ; 
request. setCharacterEncoding( " GB2312") ; 
PrintWriter out= response. getWriter() ; 
HttpSession session request. getSession(true) ; 
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String save login— request. getParameter("save login"); 
String user name— request. getParameter("user name"); 
String user password request. getParameter("user password"); 
// 用 户 登录 验证 
UserDataBean udb 一 new UserDataBean() ; 
UserBean userBean= new UserBean() ; 
userBean.setUser Name(user name); 
userBean. setUser. Password(user password); 
boolean result- udb. loginUser(userBean) ; 
ifCresult) ( 
// 处 理 cookie 
String str_user_name = URLEncoder. encode(user_name); 
Cookie user name cookie = new Cookie("icefish user name", str user name); 
Cookie user password cookie — new Cookie("icefish user password", user password); 
if(save login = null) | 
) 
else if(save login. equals("no. time"))( 
user name cookie. setMaxAge(0) ; 
user password cookie. setMaxAge(0) ; 
response. addCookie(user name cookie); 
response. addCookie(user password cookie); 
) 
else if(save login. equals("one day"))1 
int oneday—60 * 60 * 24; 
user name cookie. setMaxAge(oneday) ; 
user password cookie. setMaxAge(oneday) ; 
response. addCookie(user name cookie); 
response. addCookie(user password cookie); 
) 
else if(save login. equals("one month"))( 
int onemonth— 60 * 60 * 24 * 31; 
user name cookie. setMaxAge(onemonth) ; 
user password cookie. setMaxAge(onemonth) ; 
response. addCookie(user name cookie) ; 
response. addCookie(user password cookie); 
) 
else if(save login. equals("one year"))( 
int oneyear— 60 * 60 * 24 * 365; 
user name cookie. setMaxAge(oneyear) ; 
user password cookie. setMaxAge(oneyear) ; 
response. addCookie(user name cookie); 
response. addCookie(user password cookie); 
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RequestDispatcher rd = request. getRequestDispatcher("index.jsp?login 一 pass&user_name 一 
"十 user_name 十 "&user_password 一 "十 user_password) ; 
rd. forward(request, response); 
)else( 
out. println("<script>alert BAM, JH P 6 sk ERR ER M11; 
this. location. href= 'index. jsp'; </script>"); 


) 
) 


12.5.3 版 块 管理 


当 管 理 员 用 户 登录 成 功 后 ,可 以 在 如 图 12-8 所 示 的 主 界面 中 选择 “论坛 后 台 管 理 ” 菜 
单 , 进 行 各 项 管理 操作 ,包括 系统 管理 ,版 块 管 
理 , 用 户 管理 .帖子 管理 ,友情 链接 管理 等 , 如 


D @ IceFish BBS © 冰 鱼 | 


© 


图 12-10 所 示 。 本 书 主要 介绍 版 块 管理 的 相关 

实现 ,包括 增加 版 块 .编辑 版 块 .删除 版 块 。 其 ©] 

他 管理 操作 与 版 块 管理 类 似 , 不 逐一 细 述 。 $ pem 
1. 视图 (JSP) (o messe 
页 面 admin addboard. jsp 提供 添加 版 块 Ji; 

的 信息 输入 界面 ; admin _ editboard. jsp 和 图 12-10 论坛 后 台 管 理 界面 


admin_delboard. jsp 分 别提 供 编辑 操作 界面 和 
删除 操作 界面 。 由 于 这 几 个 页 面 代 码 主体 很 相似 ,所 以 在 此 只 列 出 admin_editboard. jsp 的 
相关 代码 ,其 运行 效果 如 图 12-11 所 示 。 


同时 腹 除 下 属 版 块 和 其 中 帖子 ! 操作 时 请 完整 填写 表单 信息 


np i 


请 仔细 阅读 说 明 ， 分 类 下 拉 荣 单 中 比 别 版面 增加 了 分 类 排序 和 分 类 复位 功能 * 
添加 新 版 块 
说 明 : 
h Ka TENGENE. WEENG ENEG BRA TING EKA YE E rp CHE PRI it, dnt 
2、 如 果 您 添加 的 是 论坛 分 类 ， 只 需要 在 所 属 分 类 中 选择 作为 论坛 分 类 即 可 ; 如 果 您 添加 的 是 论坛 版 面 , 则 要 在 所 属 分 类 中 确 ; 


论坛 名 称 Game 
f KNA 
版 面 说 明 
可 以 使 用 HDL 代码 
Li eis NGEH TRI] 
| 注意 填写 正确 的 版 主 名 称 ， 否 风 所 设 的 版 主将 无 效 tt 
添加 版 块 


图 12-11 添加 版 块 界面 
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将 全 部 信息 输入 完毕 , 单 击 “ 添 加 版 块 ”按钮 成 功 添加 后 ,就 可 以 在 论坛 版 块 中 查看 到 新 
增加 的 “Game” 版 块 ,并 且 可 以 在 此 基础 上 对 其 进行 编辑 版 块 和 删除 版 块 的 操作 ,如 图 12-12 
所 示 。 单 击 “ 编 辑 该 版 块 ? 选 项 将 回 到 图 12-11 所 示 的 界面 ,可 以 对 版 块 的 相关 信息 进行 再 
次 编辑 ; 单 击 “ 删 除 该 版 块 ?选项 将 会 删除 “Game” 版 块 。 


Ë Game 编辑 该 版 块 Wick 


12-12 新 增 版 块 操作 选项 
代码 模板 admin_addboard. jsp 4H F : 


<% Q page pageEncoding= "gb2312" Ya > 
<% (8 page contentType- "text/html;charset— gb2312" $47» 
<% (3 include file— "admin top.jsp" 947» 
<% Q& page import— "java. sql. * "967 
—jsp:useBean id= "indexBean" scope— "page" class= "net. icefish. bean. IndexBean" /> 
<html> 
<body> 
<table cellpadding=3 cellspacing=1 align=center class= "tableBorder" style= "width:96 % "> 
<tr align= center> 
<td width="100%" height=25 colspan=2 background 一 ". . /images/skin/1/bg td. gif"> 
<font color=" & ffffff"— <b> Ji Ht Fl — / b» </font> 
</td> 
</tr> 
<tr> 
<td width="100%" class— "forumRowHighlight" colspan 一 2 二 一 b> 注 意 : </b> 
到 br>@ 删 除 版 块 同时 将 删除 该 版 块 下 所 有 帖子 ! 删 除 分 类 同时 删除 下 属 版 块 和 其 中 帖子 ! 操 
作 时 请 完整 填写 表单 信息 .二 br>@ 如 果 选 择 复位 所 有 版 面 , 则 所 有 版 面 都 将 作为 一 级 论坛 (分 类 )， 
这 时 您 需要 重新 对 各 个 版 面 进行 归属 的 基本 设置 ,不 要 轻易 使 用 该 功能 , 仅 在 做 出 了 错误 的 设置 而 
无 法 复原 版 面 之 间 的 关系 和 排序 的 时 候 使 用 ,在 这 里 您 也 可 以 只 针对 某 个 分 类 进行 复位 操作 ( 见 分 
类 的 更 多 操作 下 拉 菜 单 ), 具 体 请 看 操作 说 明 . 
<br><font color=" # 0000ff" 盖 每 个 版 面 的 更 多 操作 请 见 下 拉 菜 单 , 操作 前 请 仔细 阅读 说 
明 , 分 类 下 拉 菜 单 中 比 别 的 版 面 增加 了 分 类 排序 和 分 类 复位 功能 .一 /font> 
</td> 
</tr> 
</table> 
<form name= "addboard" method="post" action=" BoardServlet?action— add" > 
<table cellpadding=3 cellspacing=1 align=center class= "tableBorder" style= "width:96 % "> 
<tr align=center> 
<td width="100%" height=25 colspan=2 background= ". . /images/skin/1/bg td. gif" > 
<font color=" # ffffff" > <b> KEMBA < /b>— /font> 
</td> 


</tr> 
<tr> 
<td width="100%" class= "forumRowHighlight" colspan— 27 ij HH : 
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去 br>1. 添 加 论坛 版 面 后 ,相关 的 设置 均 为 默认 设置 ,请 返回 论坛 版 面 管理 首页 版 面 列表 
的 高 级 设置 中 设置 该 论坛 的 相应 属性 , 如果 您 想 对 该 论坛 做 更 具体 的 权限 设置 ,请 到 论坛 权限 管理 
中 设置 相应 用 户 组 在 该 版 面 的 权限 .二 br 二 2. <font color=" # 0000ff" > wm JR fis is JI BJ Je i6 ka 43 28 
二/font>, 只 需要 在 所 属 分 类 中 选择 作为 论坛 分 类 即 可 ; 
<font color=" # 0000ff" > tn R fé: is Jn A Je i& Tz: LII — / font» , 
则 要 在 所 属 分 类 中 确定 并 选择 该 论坛 版 面 的 上 级 版 面 
</td> 
</tr> 
<tr> 
<td class= "Forumrow" width= "4026" i&3z 44 PR </td> 
<td class= "Forumrow" width= "60% "> 
<input type="text" name= "board name" size— "26"> 
</td> 
</tr> 
<tr> 
<td class= "Forumrow" width= "40% "> fi H i6 BH <br> n] E HTML 代码 一 /td> 
<td class="Forumrow" width= "60% "> 
< textarea rows="4" name= "board_info" cols="31"></textarea> 
</td> 
</tr> 
<tr> 
<td class= "Forumrow" width= "40% "> Bf Ji Mk </td> 
<td class= "Forumrow" width="60%"> 
<select name= "board bid" — option value=" "> WIZA% — R 
<% 
String sql3= "select * from icefish_board WHERE board_isMother= 'true' order by 
board_id asc"; 
ResultSet rs3= null; 
rs3= indexBean. executeSQL(sql3) ; 
while(rs3. next()){%> 
<option value= "< % =rs3. getŠtring( "board_id")⁄ >"> 
<% — rs3. getString( "board name") Ya > 
<% 
} 
rs3. close); 
indexBean. close) ; > 
«select — / td 
</tr> 
<tr> 
<td class= "Forumrow" width= "40% "之 论坛 版 主 
<br > EERE IE 86 P i E An , # WJ Br DE P Ik ECC 
</td> 
<td class= "Forumrow" width= "60% "> 
< input type="text" name= "board master" size— "26"— 
</td> 
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</tr> 
<tr> 
<td class= "Forumrow" width= "40% "></td> 


<td class= "Forumrow" width="60% "> 
<input type="submit" value 二 "添加 版 块 " onclick— " AddBoardO "> 
</td> 
</tr> 
</table> 
</form> 
</body> 
<% @include file— "admin foot.jsp" %> 
</html> 


2. 控制 器 (Servlet) 

该 控制 器 Servlet 对 象 的 名 称 是 BoardServlet( 见 12. 3. 4 WW web. xml 配置 文件 )。 控 
制 器 获取 视图 的 请 求 后 ,将 视图 中 的 信息 封装 在 实体 模型 BoardBean( 见 12. 4. 2 节 ) 中 , 然 
后 调用 业务 模型 BoardDataBean ( W, 12. 4. 3 节 ) 中 的 方法 执行 添加 、 修 改 和 删除 的 业务 
处 理 。 

代码 模板 BoardServlet. java 如 下 : 


package net.icefish. servlet; 
import net. icefish. bean. * ; 
import java. io. IOException; 
import java. io. PrintWriter; 
import javax. servlet. * ; 
import javax. servlet. http. * ; 
public class BoardServlet extends HttpServlet { 
public void init(ServletConfig config) throws ServletException ( 
super. init(config) ; 
) 
public void destroy() { 
super. destroy() ; 
) 
protected void doGet( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
doPost(request, response); 


protected void doPost( HttpServletRequest request, 
HttpServletResponse response) throws ServletException, IOException ( 
response. setContentType( "text/html ;charset— GB2312") ; 
request. setCharacterEncoding(" GB2312") ; 
PrintWriter out= response. getWriter() ; 
String action= request. getParameter( "action" ) ; 
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BoardDataBean bdb— new BoardDataBean() ; 
BoardBean bb— new BoardBean() ; 
ifCaction. equals("add")) í 


) 


String board name- request. getParameter( "board name"); 
String board info— request. getParameter("board info"); 
String board master request. getParameter("board master") ; 
String board bid— request. getParameter("board bid"); 
if(board bid. equals(""))( 

bb.setBoard Name(board name); 

bb.setBoard Info(board info) ; 

bb.setBoard Master(board master) ; 

bb.setBoard IsMother(true) ; 

boolean result— bdb. addBoard(bb) ; 

ifCresult) ( 

out. println( " script" alert( 25 JI 3r WR HE 9,3] ! ') ; 
location. href= 'admin board. jsp' ; </script>"); 
}else{ 


out. println("<script>alert( ' 添 加 新 版 块 失败 ! 请 输入 正确 的 信息 再 点 击 添加 ."); 


window. location. reload() ; </script>"); 
) 
Jelse( 
bb.setBoard Name(board name); 
bb.setBoard Info(board info) ; 
bb.setBoard Master(board master) ; 
bb.setBoard IsMother( false) ; 
bb.setBoard BID(board bid) ; 
boolean result— bdb. addBoard(bb) ; 
if(result) ( 
out. println(" 过 script>alert( ' 添 加 新 版 块 成 功 !'); 
location. href= ‘admin_board. jsp';</script>"); 
}else{ 


out. println("<script>alert( ' 添 加 新 版 块 失败 ! 请 输入 正确 的 信息 再 点 击 添加 . ); 


window. location. reload() ; </script>"); 


else if(action. equals( "edit" )) ( 


String board id— request. getParameter("board id"); 
String board name- request. getParameter("board name"); 
String board info— request. getParameter("board info"); 
String board master— request. getParameter("board master") ; 
String board bid— request. getParameter("board bid"); 
if(board bid. equals(""))( 

bb.setBoard ID(board id) ; 
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bb.setBoard Name(board name); 
bb.setBoard Info(board info); 
bb.setBoard Master(board master); 
bb.setBoard IsMother(true) ; 
boolean result— bdb. editBoard(bb) ; 
if(result) ( 

out. println( " script» alert( "修改 版 块 成 功 !"); 

location. href= 'admin_board. jsp'; </script>"); 

}else{ 

out. println( "<script> alert (f Bh He A W Vif h A IE WA B fri BERGE SI) ; 


window. location. reload() ; </script>"); 


bb.setBoard ID(board id) ; 
bb.setBoard Name(board name); 
bb.setBoard Info(board info) ; 
bb.setBoard Master(board master) ; 
bb.setBoard IsMother(false) ; 
bb.setBoard BID(board bid); 
boolean result— bdb. editBoard(bb) ; 
if(result) ( 

out. println( " script alert( MUNI |); 

location. href= 'admin board. jsp'; </script>"); 

}else{ 

out. println( " script alert( "修改 版 块 失 败 ! 请 输入 正确 的 信息 再 点 击 添 加 .'); 


window. location. reload() ; </script>"); 


) 
else if(action. equals( "del" )) { 
String board_id= request. getParameter("board id"); 
String board isMother— request. getParameter("board isMother") ; 
if(board isMother. equals("true")) ( 
bb.setBoard IsMother(true) ; 
bb.setBoard ID(board id); 
boolean result— bdb. delBoard(bb) ; 
ifCresult) ( 
out. println( " script» alert( Bl Ri E 1) ; 
location. href= 'admin board. jsp'; </script>"); 
}else{ 
out. println("<script>alert ' 删 除 版 块 失败 ! 请 再 试 一 次 . '); 
window. location. reload() ; </script>"); 
} 
)else( 
bb.setBoard_IsMother(false) ; 
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bb.setBoard ID(board id) ; 
boolean result— bdb. delBoard( bb) ; 
if(result) ( 
out. println( "< script>alert( "删除 版 块 成 功 ! ); 
location. href= 'admin_board. jsp';</script>"); 
}else{ 
out. println( "<script>2>alert( "删除 版 块 失败 ! 请 再 试 一 次 . ; 


window. location. reload() ; </script>"); 


附录 A 


HTML 常用 标签 


表 A-1 是 HTML 常用 标签 一 览 表 。 
KAI HTML 标签 一 览 表 


标 = 描 g 
c mm M sq 定义 注释 
<! DOCTYPE> 定义 文档 类 型 
<a> 定义 锚 
<abbr> 定义 缩写 
acronym 定义 只 取 首 字母 的 缩写 
<address> 定义 文档 作者 或 拥有 者 的 联系 信息 
<applet> 不 赞成 使 用 。 定 义 嵌 入 的 applet 
<area> 定义 图 像 映 射 内 部 的 区 域 
<article> 定义 文章 
<aside> 定义 页 面 内 容 之 外 的 内 容 
<audio> 定义 声音 内 容 
<b> 定义 粗 体 字 
<base> 定义 页 面 中 所 有 链接 的 默认 地 址 或 默认 目标 
<basefont> 不 赞成 使 用 。 定 义 页 面 中 文本 的 默认 字体 、 颜 色 或 尺寸 
<bdi> 定义 文本 的 文本 方向 ,使 其 脱离 其 周围 文本 的 方向 设置 
<bdo> 定义 文字 方向 
<big> 定义 大 号 文本 
<blockquote> 定义 长 的 引用 
<body> 定义 文档 的 主体 
<br> 定义 简单 的 折 行 
<button> 定义 按钮 (push button) 
<canvas> 定义 图 形 
<caption> 定义 表格 标题 
<center> 不 赞成 使 用 。 定 义 居中 文本 
<cite> 定义 引用 (citation) 
<code> 定义 计算 机 代码 文本 
«col 定义 表格 中 一 个 或 多 个 列 的 属性 值 
<colgroup> 定义 表格 中 供 格式 化 的 列 组 
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续 表 
标 签 描 述 
<command> 定义 命令 按钮 
<datalist> 定义 下 拉 列 表 
<dd> 定义 列表 中 项 目的 描述 
<del> 定义 被 删除 文本 
<details> 定义 元 素 的 细节 
<dir> 不 赞成 使 用 。 定 义 目录 列表 
<div> 定义 文档 中 的 节 
<dfn> 定义 项 目 
<dialog> 定义 对 话 框 或 窗口 
<dl> 定义 列表 
<dt> 定义 列表 中 的 项 目 
<em> 定义 强调 文本 
<embed> 定义 外 部 交互 内 容 或 插件 
—fieldset— 定义 围绕 表单 中 元 素 的 边框 
—figcaption2 定义 figure 元 素 的 标题 
<figure> 定义 媒介 内 容 的 分 组 ,以 及 它们 的 标题 
<font> 不 赞成 使 用 。 定 义 文字 的 字体 、 尺 寸 和 颜色 
<footer> 定义 section 或 page 的 页 脚 
<form> 定义 供用 户 输入 的 HTML 表单 
frame 定义 框架 集 的 窗口 或 框架 
<frameset> 定义 框架 集 
<h1> to <h6> 定义 HTML 标题 
<head> 定义 关于 文档 的 信息 
— header XE X. section 或 page 的 页 眉 
<hr> 定义 水 平 线 
<html> 定义 HTML 文档 
<i> 定义 斜体 字 
<iframe> 定义 内 联 框 架 
<img> 定义 图 像 
<input> 定义 输入 控件 
<ins> 定义 被 插入 文本 
<isindex> 不 赞成 使 用 。 定 义 与 文档 相关 的 可 搜索 索引 
<kbd> 定义 键盘 文本 
<keygen> 定义 生成 密 钥 
«label 定义 input 元 素 的 标注 
<legend> 定义 fieldset 元 素 的 标题 
<li> 定义 列表 的 项 目 
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S 
标 签 描 述 
<link> 定义 文档 与 外 部 资源 的 关系 
<map> 定义 图 像 映 射 
mark 定义 有 记号 的 文本 
<menu> 定义 命令 的 列表 或 菜单 
<menuitem> 定义 用 户 可 以 从 弹出 菜单 调用 的 命令 /菜单 项 目 
<meta> 定义 关于 HTML 文档 的 元 信息 
<meter> 定义 预定 义 范围 内 的 度量 
<noframes> 定义 针对 不 支持 框架 的 用 户 的 蔡 代 内 容 
<noscript> 定义 针对 不 支持 客户 端 脚本 的 用 户 的 蔡 代 内 容 
<object> 定义 内 嵌 对 象 
<ol> 定义 有 序列 表 
<optgroup> 定义 选择 列表 中 相关 选项 的 组 合 
<option> 定义 选择 列表 中 的 选项 
<output> 定义 输出 的 一 些 类 型 
<p> 定义 段落 
<param> 定义 对 象 的 参数 
<pre> 定义 预 格式 文本 
< progress> 定义 任何 类 型 的 任务 的 进度 
<q> 定义 短 的 引用 
<rp> 定义 若 浏览 器 不 支持 ruby 元 素 显示 的 内 容 
<rt> 定义 ruby 注释 的 解释 
«ruby 定义 ruby 注释 
«s 不 赞成 使 用 。 定 义 加 删除 线 的 文本 
<samp> 定义 计算 机 代码 样本 
<script> 定义 客户 端 脚本 
section? 定义 section 
select 定义 选择 列表 (下 拉 列 表 ) 
<small> 定义 小 号 文本 
<source> 定义 媒介 源 
<span> 定义 文档 中 的 节 
<strike> 不 赞成 使 用 。 定 义 加 删除 线 文本 
<strong> 定义 强调 文本 
<style> 定义 文档 的 样式 信息 
<sub> 定义 下 标 文本 
<summary> 为 «details 元 素 定义 可 见 的 标题 
<sup> 定义 上 标 文 本 
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标 签 描 述 
<table> 定义 表格 
<tbody> 定义 表格 中 的 主体 内 容 
<td> 定义 表格 中 的 单元 
<textarea> 定义 多 行 的 文本 输入 控件 
«toot 定义 表格 中 的 表 注 内 容 ( 脚 注 ) 
<th> 定义 表格 中 的 表 头 单元 格 
<thead> 定义 表格 中 的 表 头 内 容 
<time> 定义 日 期 /时 间 
<title> 定义 文档 的 标题 
<tr> 定义 表格 中 的 行 
track 定义 用 在 媒体 播放 器 中 的 文本 轨道 
xu 定义 打字 机 文本 
<u> 不 赞成 使 用 。 定 义 下 划 线 文本 
<u> 定义 无 序列 表 
<var> 定义 文本 的 变量 部 分 
<video> 定义 视频 
<wbr> 定义 转换 行 
<xmp> 不 赞成 使 用 。 定 义 预 格式 文本 


HRB 
HTML 中 的 颜色 表示 


d B-1 是 HTML 中 使 用 的 颜色 值 。 
表 B-1 HTML 颜色 列表 


B 色 名 十 六 进 制 颜色 值 
AliceBlue # FOF8FF 
AntiqueWhite # FAEBD7 
Aqua #00FFFF 
Aquamarine #7FFFD4 
Azure # FOFFFF 
Beige # F5F5DC 
Bisque # FFE4C4 
Black # 000000 
BlanchedAlmond # FFEBCD 
Blue #0000FF 
BlueViolet #8A2BE2 
Brown #A52A2A 
BurlyWood # DEB887 
CadetBlue * 5F9EAO 
Chartreuse * 7FFF00 
Chocolate * D2691E 
Coral # FF7F50 
CornflowerBlue * 6495ED 
Cornsilk # FFF8DC 
Crimson # DC143C 
Cyan #00FFFF 
DarkBlue #00008B 
DarkCyan #008B8B 
DarkGoldenRod # B8860B 
DarkGray #A9A9A9 
DarkGreen #006400 
DarkKhaki # BDB76B 
DarkMagenta #8B008B 
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续 表 

颜 色 名 十 六 进 制 颜色 值 
DarkOliveGreen #556B2F 
Darkorange # FF8C00 
DarkOrchid #9932CC 
DarkRed #8B0000 
DarkSalmon # E9967A 
DarkSeaGreen # 8FBC8F 
DarkSlateBlue # 483D8B 
DarkSlateGray #2F4F4F 
DarkTurquoise #00CED1 
DarkViolet # 9400D3 
DeepPink # FF1493 
DeepSkyBlue #00BFFF 
DimGray # 696969 
DodgerBlue #1E90FF 
Feldspar # D19275 
FireBrick # B22222 
FloralWhite # FFFAF0 
ForestGreen #228B22 
Fuchsia # FF00FF 
Gainsboro # DCDCDC 
Ghost White # F8F8FF 
Gold # FFD700 
GoldenRod *DAA520 
Gray # 808080 
Green # 008000 
GreenYellow # ADFF2F 
HoneyDew # FOFFFO 
HotPink # FF69B4 
IndianRed * CD5C5C 
Indigo #4B0082 
Ivory # FFFFF0 
Khaki # FOE68C 
Lavender * E6E6FA 
LavenderBlush # FFFOF5 
LawnGreen # 7CFC00 
LemonChiffon # FFFACD 
LightBlue # ADD8E6 
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续 表 
颜 色 名 十 六 进 制 颜色 值 

LightCoral # F08080 
LightCyan # EOFFFF 
LightGoldenRodYellow # FAFAD2 
LightGrey # D3D3D3 
LightGreen #90EE90 
LightPink # FFB6C1 
LightSalmon #FFA07A 
LightSeaGreen #20B2AA 
LightSkyBlue #87CEFA 
LightSlateBlue #8470FF 
LightSlateGray #778899 
LightSteelBlue # BOC4DE 
LightYellow # FFFFE0 
Lime # 00FF00 
LimeGreen # 32CD32 
Linen # FAFOE6 
Magenta # FF00FF 
Maroon # 800000 
MediumAquaMarine #66CDAA 
MediumBlue # 0000CD 
MediumOrchid # BA55D3 
MediumPurple #9370D8 
MediumSeaGreen #3CB371 
MediumSlateBlue #7B68EE 
MediumSpringGreen #00FA9A 
MediumTurquoise #48D1CC 
MediumVioletRed #C71585 
MidnightBlue #191970 
MintCream # F5FFFA 
MistyRose # FFE4E1 
Moccasin # FFE4B5 
NavajoWhite # FFDEAD 
Navy # 000080 
OldLace # FDF5E6 
Olive # 808000 
OliveDrab #6B8E23 
Orange * FFA500 
OrangeRed # FF4500 
Orchid # DA70D6 
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续 表 
颜 色 名 十 六 进 制 颜色 值 

PaleGoldenRod 井 EEE8AA 
PaleGreen # 98FB98 
PaleTurquoise * AFEEEE 
PaleVioletRed # D87093 
PapayaWhip # FFEFD5 
PeachPuff # FFDAB9 
Peru # CD853F 
Pink # FFCOCB 
Plum # DDAODD 
PowderBlue # BOEOE6 
Purple # 800080 
Red # FF0000 
RosyBrown # BC8F8F 
RoyalBlue #4169E1 
SaddleBrown * 8B4513 
Salmon # FA8072 
SandyBrown # FAA460 
SeaGreen # 2E8B57 
SeaShell # FFF5EE 
Sienna # A0522D 
Silver # COCOCO 
SkyBlue 4 87CEEB 
SlateBlue S 6A5ACD 
SlateGray # 708090 
Snow # FFFAFA 
SpringGreen #00FF7F 
SteelBlue # 4682B4 
Tan # D2B48C 
Teal # 008080 
Thistle # D8BFD8 
Tomato # FF6347 
Turquoise # 40E0DO 
Violet # EE82EE 
VioletRed # D02090 
Wheat # F5DEB3 
White # FFFFFF 
WhiteSmoke #F5F5F5 
Yellow # FFFF00 
YellowGreen * 9ACD32 
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1. request 对 象 

客户 端的 请 求 信息 被 封装 在 request 对 象 中 ,通过 它 才 能 了 解 到 客户 的 需求 ,然后 做 出 
响应 。 它 是 HttpServletRequest 类 的 实例 。 其 方法 说 明 如 下 : 

(1) object getAttribute(String name) 返回 指定 属性 的 属性 值 。 

(2) Enumeration getAttributeNames() 返回 所 有 可 用 属性 名 的 枚 举 。 

(3) String getCharacterEncoding() 返回 字符 编码 方 。 

(4) int getContentLength() 返回 请 求 体 的 长 度 (以 字 节 数 ) 。 

(5) String getContentType() 得 到 请 求 体 的 MIME 类 型 。 

(6) ServletInputStream getInputStream() 得 到 请 求 体 中 一 行 的 二 进 制 流 。 

(7) String getParameter(String name) 返回 name 指定 参数 的 参数 值 。 

(8) Enumeration getParameterNames() 返回 可 用 参数 名 的 枚 举 。 

(9) String[] getParameterValues(String name) 返回 包含 参数 name 的 所 有 值 的 数组 。 

(10) String getProtocol() 返回 请 求 用 的 协议 类 型 及 版 本 号 。 

(1D String getSchemeO 返回 请 求 用 的 计划 名 ,如 http、https 及 ftp 等 。 

(12) String getServerName() 返回 接受 请 求 的 服务 器 主机 名 。 

(13) int getServerPort() 返回 服务 器 接受 此 请 求 所 用 的 端口 号 。 

(14) BufferedReader getReader() 返回 解码 过 的 请 求 体 。 

(15) String getRemoteAddr() 返回 发 送 此 请 求 的 客户 端 IP 地 址 。 

(16) String getRemoteHost() 返回 发 送 此 请 求 的 客户 端 主机 名 。 

(17) void setAttribute(String key, Object obj) 设置 属性 的 属性 值 。 

(18) String getRealPath(String path) 返回 一 个 虚拟 路 径 的 真实 路 径 。 

(19) void setCharacterEncoding(String code) 设 置 统一 字符 编码 。 

2. response 对 象 

response 对 象 包含 了 响应 客户 请 求 的 有 关 信 息 , 但 在 JSP 中 很 少 直接 用 到 它 。 它 是 
HttpServletResponse 类 的 实例 。 其 方法 说 明 如 下 : 

(D String getCharacterEncoding() 返回 响应 用 的 是 何 种 字符 编码 。 

(2) ServletOutputStream getOutputStream() 返回 响应 的 一 个 二 进 制 输出 流 。 

(3) Print Writer getWriter() 返回 可 以 向 客户 端 输出 字符 的 一 个 对 象 。 
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(4) void setContentLength(int len) 设置 响应 头 长 度 。 

(5) void setContentType(String type) 设置 响应 的 MIME 类 型 。 

(6) void set Header(String name.String value) 设置 指定 名 字 的 HTTP 文件 头 的 值 ， 
以 此 来 操作 HTTP 文件 头 。 

(7) sendRedirect(java. lang. String location) 重新 定向 客户 端的 请 求 。 

3. session 对 象 

session 对 象 是 HttpSession 类 的 实例 , 指 的 是 客户 端 与 服务 器 的 一 次 会 话 , 从 客户 连 到 
服务 器 的 一 个 WebApplication 开始 ,直到 客户 端 与 服务 器 断 开 连接 为 止 。 其 方法 说 明 
如 下 : 

(D long getCreationTime() 返回 session 创建 时 间 。 

(2) public String getId() 返回 session 创建 时 JSP 引擎 为 它 设 的 唯一 ID 号 。 

(3) long getLastAccessedTime() 返回 此 session 里 客户 端 最 近 一 次 请 求 时 间 。 

(4) int getMaxlnactivelnterval() 返回 两 次 请 求 间 隔 多 长 时 间 此 session 被 取消 (ms)。 

(5) String[ ] getValueNames() 返回 一 个 包含 此 session 中 所 有 可 用 属性 的 数组 。 

(6) void invalidateO 取消 session, ffi session 不 可 用 。 

(7) boolean isNew() 返回 服务 器 创建 的 一 个 session ,客户 端 是 否 已 经 加 入 。 

(8) void removeValue(String name) 删除 session 中 指定 的 属性 。 

(9) void setMaxInactiveInterval() 设置 两 次 请 求 间隔 多 长 时 间 此 session 被 取消 (ms)。 

(10) getAttribute(String name) 获取 session 属性 值 。 

(11) setAttribute(String name) 设置 session 属性 值 。 

4. Out 对 象 

out 对 象 是 JspWriter 类 的 实例 ,是 向 客户 端 输出 内 容 常 用 的 对 象 。 其 方法 说 明 如 下 : 

(1) void clear() 清除 缓冲 区 的 内 容 。 

(2) void clearBuffer() 清除 缓冲 区 的 当前 内 容 。 

(3) void flushO 清空 流 。 

(4) int getBufferSize() 返回 缓冲 区 以 字 节 数 的 大 小 ,如 不 设 缓冲 区 则 为 0。 

(5) int getRemaining() 返回 缓冲 区 还 剩余 多 少 可 用 。 

(6) boolean isAutoFlush() 返回 缓冲 区 满 时 ,是 自动 清空 还 是 抛 出 异常 。 

(7) void close) 关闭 输出 流 。 

5. page 对 象 

page 对 象 就 是 指向 当前 JSP 页 面 本 身 , 有 点 像 类 中 的 this 指针 , 它 是 java. lang. Object 
类 的 实例 。 其 方法 说 明 如 下 : 

(1) class getClass 返回 此 Object 的 类 。 

(2) int hashCodeO 返回 此 Object 的 hash 码 。 

(3) boolean equals(Object obj) 判断 此 Object 是 否 与 指定 的 Object 对 象 相等 。 
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(4) void copyCObject obj) 把 此 Object 复制 到 指定 的 Object 对 象 中 。 

(5) Object clone() 克隆 此 Object 对 象 。 

(6) String toString() 把 此 Object 对 象 转换 成 String 类 的 对 象 。 

(7) void notify() 唤醒 一 个 等 待 的 线程 。 

(8) void notifyAll() 唤醒 所 有 等 待 的 线程 。 

(9) void wait(int timeout) 使 一 个 线程 处 于 等 待 直到 timeout 结束 或 被 唤醒 。 

(10) void wait() 使 一 个 线程 处 于 等 待 直 到 被 唤醒 。 

(1D void enterMonitor() Xf Object 加 锁 。 

(12) void exitMonitor() 对 Object 开锁 。 

6. application 对 象 

application 对 象 实现 了 用 户 间 数据 的 共享 ,可 存放 全 局 变量 。 它 开始 于 服务 器 的 启动 ， 
直到 服务 器 的 关闭 ,在 此 期 间 , 此 对 象 将 一 直 存在 ; 这 样 在 用 户 的 前 后 连接 或 不 同 用 户 之 间 
的 连接 中 ,可 以 对 此 对 象 的 同一 属性 进行 操作 ; 在 任何 地 方 对 此 对 象 属性 的 操作 ,都 将 影响 
到 其 他 用 户 对 此 的 访问 。 服 务 器 的 启动 和 关闭 决定 了 application 对 象 的 生命 。 它 是 
ServletContext 类 的 实例 。 其 方法 说 明 如 下 : 

(1) Object getAttribute( String name) 返回 给 定名 的 属性 。 

(2) Enumeration getAttributeNames() 返回 所 有 可 用 属性 名 的 枚 举 。 

(3) void setAttribute(String name, Object obj) 设 定 属性 的 属性 值 。 

(4) void removeAttribute(String name) 删除 一 个 属性 及 其 属性 值 。 

(5) String getServerInfoO 返回 JSP(SERVLET) 引 擎 名 及 版 本 号 。 

(6) String getRealPath(String path) 返回 一 虚拟 路 径 的 真实 路 径 。 

(7) ServletContext getContext ( String uripath) 返回 指定 WebApplication 的 
application 对 象 。 

(8) int getMajorVersion() 返回 服务 器 支持 的 Servlet API 的 最 大 版 本 号 。 

(9) int getMinorVersion() 返回 服务 器 支持 的 Servlet API 的 最 小 版 本 号 。 

(10) String getMimeType(String file) 返回 指定 文件 的 MIME 类 型 。 

(11) URL getResource(String path) 返回 指定 资源 (文件 及 目录 ) 的 URL 路 径 。 

(12) InputStream getResourceAsStream(String path) 返回 指定 资源 的 输入 流 。 

(13) RequestDispatcher getRequestDispatcher (String uripath) 返回 指定 资源 的 
RequestDispatcher 对 象 。 

(14) Servlet getServlet(String name) 返回 指定 名 的 Servlet。 

(15) Enumeration getServlets() 返回 所 有 Servlet 的 枚 举 。 

(16) Enumeration getServletNames() 返回 所 有 Servlet 名 的 枚 举 。 

(17) void log(String msg) 把 指定 消息 写 入 Servlet 的 日 志文 件 。 

(18) void logCException exception.String msg) 把 指定 异常 的 栈 轨 迹 及 错误 消息 写 人 
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Servlet 的 日 志文 件 。 

(19) void log(String msg, Throwable throwable) 把 栈 轨迹 及 给 出 的 Throwable 异常 
的 说 明 信 息 写 入 Servlet 的 日 志文 件 。 

7. exception 对 象 

exception 对 象 是 一 个 例外 对 象 , 当 一 个 页 面 在 运行 过 程 中 发 生 了 例外 ,就 产生 这 个 对 
象 。 如 果 一 个 JSP 页 面 要 应 用 此 对 象 , 就 必须 把 isErrorPage 设 为 true, 否 则 无 法 编译 。 它 
实际 上 是 java. lang. Throwable 的 对 象 。 其 方法 说 明 如 下 : 

(1) String getMessage() 返回 描述 异常 的 消息 。 

(2) String toString() 返回 关于 异常 的 简短 描述 消息 。 

(3) void printStackTrace() 显示 异常 及 其 栈 轨迹 。 

(4) Throwable Fill StackTrace() 重 写 异常 的 执行 栈 轨 迹 。 

8. pageContext 对 象 

pageContext 对 象 提供 了 对 JSP 页 面 内 所 有 的 对 象 及 名 字 空 间 的 访问 ,也 就 是 说 , 它 可 
以 访问 到 本 页 所 在 的 session ,也 可 以 取 本 页 面 所 在 的 application 的 某 一 属性 值 , 它 相当 于 
页 面 中 所 有 功能 的 集大成 者 , 它 的 本 类 名 也 叫 pageContext。 其 方法 说 明 如 下 : 

CD JspWriter getOut() 返回 当前 客户 端 响应 被 使 用 的 JspWriter i Cout) 。 

(2) HttpSession getSession() 返回 当前 页 中 的 HttpSession XF# (session) 。 

(3) Object getPageO 返回 当前 页 的 Object 对 象 (page) 。 

(4) ServletRequest getRequestO 返回 当前 页 的 ServletRequest 对 象 (request) 。 

(5) ServletResponse getResponseO 返回 当前 页 的 ServletResponse 对 象 (response) 。 

(6) Exception getException() 返回 当前 页 的 Exception Xf # (exception) 。 

(7) ServletConfig getServletConfigO 返回 当前 页 的 ServletConfig XJ # (config) 。 

(8) ServletContext getServletContext() 返回 当前 页 的 ServletContext 对 象 (application) 。 

(9) void setAttribute(String name. Object attribute) 设置 属性 及 属性 值 。 

(10) void setAttribute(String name. Object obj ,int scope) 在 指定 范围 内 设置 属性 及 
属性 值 。 

(11) public Object getAttribute(String name) 取 属 性 的 值 。 

(12) Object getAttribute(String name.int scope) 在 指定 范围 内 取 属 性 的 值 。 

(13) public Object findAttribute(String name) 寻找 一 属性 ,返回 其 属性 值 或 NULL, 

(14) void removeAttribute(String name) 删除 某 属 性 。 

(15) void removeAttribute(String name.int scope) 在 指定 范围 删除 某 属 性 。 

(16) int getAttributeScope(String name) 返回 某 属性 的 作用 范围 。 

(17) Enumeration getAttributeNameslnScope(int scope) 返回 指定 范围 内 可 用 的 属性 
名 枚 举 。 

(18) void release() 释放 pageContext 所 占用 的 资源 。 
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(19) void forward(String relativeUrlPath) 使 当前 页 面 重 导 到 另 一 页 面 。 

(20) void include(String relativeUrlPath) 在 当前 位 置 包含 另 一 文件 。 

9. config 对 象 

config 对 象 是 在 一 个 Servlet 初始 化 时 ,JSP 引擎 向 它 传递 信息 用 的 ,此 信息 包括 
Servlet 初始 化 时 所 要 用 到 的 参数 (通过 属性 名 和 属性 值 构成 ) 以 及 服务 器 的 有 关 信息 (通过 
传递 一 个 ServletContext 对 象 ) 。 将 其 方法 说 明 如 下 : 

(1) ServletContext getServletContext() 返回 含有 服务 器 相关 信息 的 ServletContext 
对 象 。 

(2) String getlnitParameter(String name) 返回 初始 化 参数 的 值 。 

(3) Enumeration getInitParameterNames() 返回 Servlet 初始 化 所 需 的 所 有 参数 的 
枚 举 。 


教学 资源 支持 
敬爱 的 教师 : 
感谢 您 一 直 以 来 对 清华 版 计算 机 教材 的 支持 和 爱护 。 为 了 配合 本 课程 的 
教学 需要 ,本 教材 配 有 配套 的 电子 教案 (素材 ) ,有 需求 的 教师 请 到 清华 大 学 出 
版 社 主 页 (http://www. tup. com. cn) 上 查询 和 下 载 ,也 可 以 拨打 电话 或 发 送 电 
子 邮 件 咨 询 。 
如 果 您 在 使 用 本 教材 的 过 程 中 遇 到 了 什么 问题 ,或 者 有 相关 教材 出 版 计 
划 , 也 请 您 发 邮件 告诉 我 们 ,以 便 我 们 更 好 地 为 您 服务 。 


我 们 的 联系 方式 : 
地 HE: 北京 海淀 区 双 清 路 学 研 大 厦 A Æ 707 
Ë £y: 100084 


ES 话 : 010— 62770175 — 4604 


课件 下 载 : http: //www. tup. com. cn 


电子 邮件 : weijjQ tup. tsinghua. edu. cn 

作者 交流 论坛 :http://itbook. kuaizhan. com/ 

教师 交流 QQ 群 : 136490705 微 信 号 : itbook8 QQ: 883604 
(申请 加 入 时 ,请 写 明 您 的 学 校 名 称 和 姓名 ) 

用 微 信 扫 一 扫 右 边 的 二 维 码 , 即 可 关注 计算 机 教材 公众 号 。 


